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1998 年 ,Michael Kunze 在 为 德国 一 家 计算 机 杂志 编写 一 篇 关于 自由 软件 如 何 成 为 
商业 软件 替代 品 的 文章 时 ,创造 了 LAMP 这 个 词 , 即 由 Linux 操作 系统 .Apache 网 络 服 
务 器 ,MySQL 数据 库 和 PHP 脚本 语言 4 种 技术 的 首 字母 组 合 而 成 。 随 之 LAMP 技术 
便 点 亮 了 自由 软件 业 的 一 慢 “ 明 灯 ” 

为 了 帮助 众多 从 事 Web 应 用 与 开发 的 读者 快速 党 所 LAMP ,提高 项 目 开发 水 平 , 笔 
者 在 多 年 从 事 LAMP 教学 及 开发 工作 的 基础 上 精心 编著 了 本 书 。 本 书 按照 由 浅 入 深 , 循 
序 渐进 的 原则 精心 组 织 各 章节 内 容 ,各 知识 点 前 后 贯穿 .但 又 自 成 体系 。 它 既 包 括 
Linux、Apache、MySQL 及 PHP 的 基础 知识 讲解 ,又 含有 综合 复杂 案例 ;使 读者 既 可 以 高 
效 地 掌握 LAMP 中 最 基础 .最 常用 的 各 项 技术 ,又 可 以 系统 地 理解 LAMP 架构 下 实际 应 
用 系统 的 完整 开发 思路 。 

本 书 为 了 简化 学 习 和 破解 PHP 结合 MySQL 应 用 项 目 开发 的 难度 , 细 分 为 11 章 。 
其 中 每 一 章节 都 专注 于 特定 的 主题 ,读者 可 以 按 主题 进行 跳跃 式 阅读 ;每 一 知识 要 点 都 紧 
密 结 合 开发 示例 ,读者 可 以 参照 示例 进行 练习 ,深刻 体会 其 中 的 要 领 。 

本 书 前 5 章 讲 解 PHP 的 基础 知识 和 基本 语法 结构 ,语言 通俗 易 懂 ,即使 是 初学 者 也 
很 容易 读 懂 并 学 会 。 第 6 章 和 第 7 章 讲解 了 PHP 结合 HTML 表单 实现 动态 程序 开发 。 
第 8 章 介绍 了 SQL 语言 基础 及 MySQL 基础 操作 。 第 9 章 简单 介绍 PHP 结合 MySQL 
开发 方法 。 第 10 章 介绍 动态 程序 开发 的 必 备 知识 一 一 cookie 和 会 话 。 此 外 ,为 了 使 读者 
能 巩固 所 学 知识 ,全 书 每 章 后 面 都 有 相应 的 实 训练 习 。 第 11 章 先 简单 介绍 了 PHP 面向 
对 象 的 开发 模式 ,然后 结合 一 个 PHP 知名 开源 框架 CodeIgniter ,详细 介绍 了 一 个 综 
合 动态 案例 的 制作 过 程 ,内 容 精 彩 、 页 面 丰富 ,是 读者 在 掌握 基础 知识 之 后 ,对 知识 进行 巩 
固 的 部 分 。 

本 书 的 实例 是 作者 从 实际 工作 中 精 选 出 来 的 ,具有 较 强 的 应 用 性 和 示范 作用 。 同 时 ， 
书 中 所 用 语言 浅显 易 懂 ,并 辅 之 以 精 选 的 配 图 ,相信 读者 只 要 按照 书 中 的 步骤 进行 操作 ， 
一 定 能 开发 出 预期 的 功能 及 效果 。 

本 书 实例 丰富 、 可 操作 性 强 , 既 可 作为 应 用 型 本 科 、 高 职高 专 院 校 职业 教育 和 继续 教 
育 的 教材 ,也 可 作为 计算 机 专业 技术 人 员 的 参考 书籍 。 

为 方便 教学 ,本 书 配 有 电子 课件 和 书 中 所 有 例子 的 代码 ,如 有 需要 ,可 至 清华 大 学 出 
版 社 网 站 下 载 。 

本 书 由 正德 职业 技术 学 院 与 钟 山 职业 技术 学 院 的 资深 教师 共同 编写 ,编者 多 年 从 事 
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第 1 章 基础 知识 


PHP 是 全 球 最 普及 、 应 用 最 广泛 的 互联 网 开发 语言 之 一 。PHP 语言 具有 简单 、 易 
学 ,源码 开放 ,可 操作 多 种 主流 与 非 主 流 的 数据 库 , 支 持 面向 对 象 的 编程 ,支持 多 种 开源 框 
架 , 支 持 跨 平台 的 操作 ,而 且 完 全 免费 等 特点 , 越 来 越 受到 广大 程序 员 的 青睐 和 认同 。 
PHP 目前 拥有 几 百 万 用 户 ,其 发 展 速度 要 快 于 在 它 之 前 的 任何 一 种 计算 机 语言 。 相 信 
PHP 一 定 能 够 经 得 起 实践 的 检验 ,发 展 成 为 互联 网 开发 语言 中 “主流 中 的 主流 ”。 


1.1 PHP、Apache、MySQL 和 开源 的 简介 


从 许多 方面 来 看 ,PHP 语言 都 是 开源 项 目的 典型 代表 ,最 初创 建 它 只 是 为 了 满足 一 
个 开发 人 员 自 己 的 需要 ,在 此 之 后 由 于 日 益 扩 大 的 PHP 社区 的 需求 而 不 断 改进 。 作 为 
一 个 刚刚 涉足 这 个 领域 的 开发 人 员 ,对 PHP 的 发 展 历程 有 所 了 解 是 很 重要 的 ,因为 它 能 
帮助 我 们 体会 到 这 种 语言 的 优势 ,另外 从 某 种 程度 上 还 可 以 理解 PHP 是 如 何 偶然 地 形 
成 其 独 有 特性 的 。 


1.1.1 开源 软件 及 其 优点 


时 至 今日 ,开源 软件 早已 成 为 软件 领域 不 可 或 缺 的 重要 组 成 部 分 。 很 多 成 功 的 开源 
软件 项 目 如 Linux、Apache、Eclipse 等 ,由 于 其 出 色 的 质量 和 固有 的 开放 性 ,被 当做 事实 
上 的 工业 标准 软件 ,广泛 地 应 用 于 各 个 领域 ,产生 了 巨大 的 社会 价值 。 

更 为 重要 的 是 ,开源 运动 宣扬 了 自由 \ 平 等 .协作 的 精神 ,实践 了 信息 和 知识 共享 的 理 
念 ,并 且 实 现 了 知识 产权 保护 和 分 享 之 间 的 微妙 平衡 。 从 这 个 意义 上 来 说 ,开源 软件 是 人 
类 对 于 理想 和 现实 权衡 之 下 的 一 个 美妙 产物 ,这 也 是 开源 运动 能 够 如 此 成 功 的 关键 原因 。 

1. 开源 软件 的 定义 

根据 开放 源码 促进 会 (Open Source Initiative.OSI) 的 定义 .可 将 开放 源码 定义 为 : 
“开放 源码 通过 支持 源 代码 的 独立 同业 互 查 (independent peer review) 和 快速 发 展演 变 提 
高 了 软件 的 可 靠 性 和 质量 。 要 通过 OSI 认证 ,软件 必须 在 获得 许可 证 的 情况 下 发 布 ,该 
许可 证 可 保证 免费 读 取 、 重 新 发 布 .修改 和 使 用 该 软件 的 权利 ,” 开 放 源 码 软 件 被 定义 为 描 
述 其 源码 可 以 被 公众 使 用 的 软件 ,并 且 此 软件 的 使 用 、 修 改 和 分 发 也 不 受 许可 证 的 限制 。 

开放 源码 促进 会 即 OSI 对 开源 软件 有 明确 的 定义 .业界 公认 只 有 符合 这 个 定义 的 软 
件 才 能 被 称 为 开放 源 代码 软件 ,简称 开源 软件 ,如 图 1-1 所 示 。 
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2. 开源 软件 和 其 他 类 型 软件 的 比较 

在 开源 运动 风起云涌 的 今天 ,开源 软件 的 触角 几乎 伸 到 了 各 个 领域 ,用 户 可 以 找到 开 
源 的 项 目 管理 软件 .开源 的 3D 电影 泻 染 引擎 ,开源 的 游戏 框架 、 开 源 的 硬件 驱动 程序 、 开 
源 的 手机 操作 系统 等 。 

Linux 操作 系统 、Apache 服务 器 、MySQL 数据 库 以 及 PHP、Perl 或 者 Python 语言 ， 
这 些 产品 共同 组 成 了 一 个 强大 的 Web 应 用 程序 平台 ,如 图 1-2 所 示 。 这 组 产品 都 是 开源 
软件 ,常用 来 搭建 动态 网 站 或 者 服务 器 ,本 身 都 是 各 自 独立 的 程序 ,但 是 因为 常 被 放 在 一 
起 使 用 ,拥有 了 越 来 越 高 的 兼容 度 ,共同 组 成 了 一 个 强大 的 Web 应 用 程序 平台 。 随 着 开 
源 潮 流 的 蓬勃 发 展 , 开 放 源 代码 的 LAMP 已 经 与 J2EE 和 . Net 商业 软件 形成 三 足 易 立 之 
势 ,并 且 该 软件 开发 的 项 目 在 软件 方面 的 投资 成 本 较 低 ,因此 受到 整个 IT 界 的 关注 。 从 
网 站 的 流量 上 来 说 ,70% 以 上 的 访问 流量 是 LAMP 提供 的 ,LAMP 是 最 强大 的 网 站 解决 
方案 。 程 序 员 在 Windows 操作 系统 下 使 用 这 些 Linux 环境 中 的 工具 称 为 使 用 WAMP。 


opensource LAM 用 


Linux Apache MySQL PHP,Perl,Python 
图 1-1 开源 标志 图 1-2 LAMP 架构 


LAMP 所 代表 的 不 仅仅 是 自由 和 开放 ,而 且 LAMP 构成 了 一 个 强大 的 、 高 性 能 Web 
应 用 平台 ,具有 易于 开发 .更 新 速度 快 、 安 全 性 高 .成 本 低 的 特点 ,因此 被 许多 开发 者 视 为 
“黄金 组 合 "。 当 前 ,国外 最 知名 的 三 大 BBS 软件 提供 商 IPB、VBB、PHPBBS 均 基 于 
LAMP 平台 。 在 国内 , 据 PHPChina 资料 统计 ,在 中 国 排名 前 200 名 的 网 站 中 就 有 61% 
采用 了 LAMP 技术 。 

LAMP 的 迅速 发 展 对 Java 和 . NET 等 商业 软件 构成 了 严重 威胁 。 据 美国 互联 网 市 
场 调 研 机 构 NetCraft(www. netcraft. co. uk) 发 布 的 2010 年 8 月 份 的 网 站 统计 数据 表明 ， 
基于 Linux 的 Apache 依然 是 网 站 的 第 一 选择 ,市 场 份额 达 半数 以 上 , 而 快速 崛起 的 
Web 2.0 网 站 ,半数 以 上 也 都 采用 了 LAMP 技术 。 


1.1.2 PHP 简介 


1. PHP 是 什么 

PHP 是 一 种 服务 器 端的 嵌入 式 脚本 语言 ,是 一 种 跨 平 台 、 面 向 对 象 .HTML 艇 入 式 
的 脚本 语言 。 它 于 1995 年 由 Rasmus Lerdorf 开发 。PHP 是 Hypertext Preprocessor 
( 超 文本 预 处 理 器 ) 的 缩写 ,是 一 种 开源 、 跨 平台 、 独 立 于 架构 、 解 释 型 .面向 对 象 、 快 速 安 
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全 ,简单 易学 性 能 优越 的 Web 服务 器 端 动态 网 页 开发 语言 。PHP 自从 推出 以 来 ,其 用 
户 数量 呈 指 数 级 增长 ,如 图 1-3 所 示 ,目前 已 有 超过 2200 万 个 网 站 、1. 5 万 家 公司 450 万 
程序 开发 人 员 在 使 用 PHP 语言 , 它 是 目前 动态 网 页 开发 中 使 用 最 为 广泛 的 语言 之 一 。 


1-3 本 图 来 自 Netcraft(www. netcraft. com) ,显示 了 过 去 几 年 PHP 的 显著 增长 


在 编写 本 书 时 ,PHP 发 行 的 稳定 版 已 到 了 版 本 5( 从 技术 上 讲 , 是 5. 3. 3) ,但 其 版 本 4 
仍然 在 使 用 并 且 在 服务 器 上 很 常见 ,PHP 的 最 新 版 本 6 也 即将 正式 发 布 。 本 书 将 使 用 
PHP5 ,但 是 ,如 果 版 本 有 差别 ,也 不 会 有 什么 问题 。 显 然 ,在 服务 器 上 最 好 是 使 用 最 新 的 
PHP 技术 ,本 书 将 尽 可 能 采用 无 版 本 差别 的 代码 。 

2. PHP 的 语言 优势 

PHP 起 源 于 自由 软件 , 即 开 放 源 代码 软件 ,使 用 PHP 进行 Web 应 用 程序 的 开发 具 
有 以 下 语言 优势 : 

(1) 安全 性 高 。PHP 是 开源 软件 ,每 个 人 都 可 以 看 到 所 有 PHP 的 源 代码 ,程序 代码 
与 Apache 编译 在 一 起 的 方式 也 可 以 让 它 具有 灵活 的 安全 设 定 ,PHP 具有 公认 的 安全 
性 能 。 

(2) 跨 平台 。PHP 几乎 支持 所 有 的 操作 系统 平台 (如 Windows 32 或 UNIX/Linux/ 
Macintosh/FreeBSD/OS2 等 ) ,并 且 支 持 Apache IIS 等 多 种 Web 服务 器 ,并 以 此 广 为 
流行 。 

(3) 支持 广泛 的 数据 库 。 可 操纵 多 种 主流 与 非 主流 的 数据 库 , 如 MySQL、Access、 
SQL Server、Oracle.DB2 等 ,其 中 PHP 与 MySQL 是 现在 最 佳 的 组 合 ,它们 的 组 合 可 以 
跨 平 台 运行 。 

(4) 简单 易学 。PHP 嵌入 在 HTML 语言 中 ,以 脚本 语言 为 主 ,内 置 丰富 函数 ,语法 
简单 .书写 容易 ,方便 学 习 掌握 。 

(5) 执行 速度 快 。 占 用 系统 资源 少 , 代 码 执行 速度 快 。 

(6) 开发 成 本 低 。 在 流行 的 企业 应 用 LAMP 平台 中 , Linux、Apache、MySQL 和 
PHP 都 是 免费 软件 ,这 种 开源 免费 的 框架 结构 可 以 为 网 站 经 营 者 节省 很 大 一 笔 开 支 。 

(7) 模板 化 。 实 现 程序 逻辑 与 用 户 界 面 分 离 。 
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(8) 支持 面向 对 象 。 支 持 面向 对 象 和 过 程 的 两 种 风格 开发 .并 可 向 下 兼容 。 面 向 对 
象 编程 (OOP) 是 当前 的 软件 开发 趋势 。PHP 对 OOP 提供 了 良好 的 支持 ,可 以 使 用 OOP 
的 思想 来 进行 PHP 的 高 级 编程 ,这 对 于 提高 PHP 编程 能 力 和 规划 好 Web 开发 架构 都 非 
常 有 意义 。 

(9) 内 内 Zend 加 速 引擎 ,性 能 稳定 快速 。 

(10) 应 用 范围 广 ,PHP 技术 在 Web 开发 的 各 个 方面 应 用 得 非常 广泛 。 世 界 上 很 多 
大 公司 都 采用 了 PHP 技术 ,如 德意志 银行 的 交易 系统 、 华 尔 街 的 股票 在 线 买卖 .汉莎 航 
空 公司 的 票务 处 理 , 其 至 美联储 ,宇航 局 都 采用 了 PHP 技术 。 

3. PHP 语言 的 发 展 

TIOBE 世界 编程 语言 排行 榜 在 一 定 程度 上 体现 了 编程 语言 在 当前 的 流行 趋势 。 
TIOBE 最 新 公布 了 2014 年 2 月 的 编程 语言 排行 榜 , 值 得 关注 的 PHP 语言 一 直 稳 居 前 
列 ,位 于 第 六 位 。2014 年 2 月 份 TIOBE 世界 编程 语言 排行 的 相关 数据 说 明 如 图 1-4 
所 示 。 


2014 年 2 月 2013 年 2 月 ”排名 变化 ”编程 语言 支持 率 ”支持 率 变化 
2 ~ C 18334% +125% 
2 1 v Jaa 17316% -107% 
3 3 ObjediveC 11341% +154% 
4 4 cr 6.892% 1.87% 
5 5 Ce 6450% 023% 
6 6 PHP 4219% -085% 
7 8 ~ Visual) Basic 2759% 1.89% 
8 v Pyhon 2157% -79% 
9 11 ~ JavaSaipt 1929% +051% 
10 12 ^~ Visual Basic NET 1798% +079% 
1 16 人 Transad-SQL 1667% +089% 
人 2 10 Y Ruby 0.924% -083% 
13 9 ¥ Per 0887% 136% 
14 18 从 MATAB 0.641% -001% 
1 22 从 PUsaL 0.604% 000% 
16 4 从 Fe 0591% +0.42% 
7 14 Y Pascal 0551% -038% 
18 36 从 D 0529% 023% 
9 13 ¥ Usp 0523% -042% 
20 全 学 DelphiObject Pascal 0522% -036% 


图 1-4 2014 年 2 月 TIOBE 世界 编程 语言 排名 


TIOBE 编程 语言 排行 榜 衡量 了 各 种 编程 语言 的 流行 程度 。 该 排行 榜 每 月 发 布 一 次 ， 
统计 数据 包括 全 球 范围 的 软件 工程 师 、 培 训 课 程 以 及 第 三 方 供应 商 ,数据 来 自 Google、 


MSN 和 Yahool! 等 流行 搜索 引擎 。 
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近 几 年 PHP 的 发 展 呈 现 上 升 趋势 ,如 图 1-5 所 示 ,这 也 说 明了 PHP 语言 简单 .易学 、 
面向 对 象 和 安全 等 特点 正在 被 更 多 人 所 认同 。 相 信 新 的 PHP 语言 将 会 朝 着 更 加 企业 化 


的 方向 迈进 ,并 且 将 更 适合 大 型 系统 的 开发 。 


PHP 语 言 走势 图 (来 源 : TIOBE) 


支持 率 (%) 


2002 2004 2006 2008 2010 
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图 1-5 PHP 近 几 年 的 语言 走势 图 


4. PHP 与 其 他 语言 的 比较 


就 目前 的 动态 网 页 开发 技术 而 言 ,除了 PHP 以 外 ,还 有 ASP、JSP 和 . NET, 它 们 各 
有 千秋 ,都 有 着 广泛 的 用 户 群 ,本 节 将 它们 进行 简单 的 比较 ,如 表 1-1 所 示 。 


表 1-1 PHP 与 其 他 语言 的 比较 


比较 项 目 PHP ASP JSP .NET 
跨 操作 系统 性 支持 只 支持 Windows 支持 只 支持 Windows 
Web 服务 器 多 IIS 很 多 TS 
执行 效率 快 快 极 快 极 快 
稳定 性 高 低 高 高 
开发 敏捷 度 高 高 中 高 
支持 语言 PHP VBScript Java C# .VB.C++ ,JScript 
函数 支持 多 少 中 多 
系统 安全 高 低 高 高 
版 本 升级 快 慢 慢 一 般 
难 易 程 度 易 易 难 中 
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它们 都 各 有 所 长 ,可 以 根据 实际 需要 从 中 选择 一 种 。 不 一 定 要 选择 最 好 的 ,但 一 定 要 
选择 最 适合 自己 的 。 


1.1.3 Apache 简介 


1. Apache 的 特性 

Apache 是 世界 排名 第 一 的 Web 服务 器 ,根据 Netcraft(www. netsraft. com) 公 司 所 
作 的 调查 ,世界 上 50% 以 上 的 Web 服务 器 在 使 用 Apache。 

1995 年 4 月, 最早 的 Apache(0.6.2 版 ) 由 Apache Group 公布 发 行 。Apache Group 
是 一 个 完全 通过 Internet 进行 运作 的 非 盈 利 机 构 , 由 它 来 决定 Apache Web 服务 器 的 标 
准 发 行 版 中 应 该 包含 哪些 内 容 。Apache 是 开源 的 。 因 此 它 准 许 任何 人 修改 隐藏 的 错误 ， 
提供 新 的 特征 和 将 它 移植 到 新 的 平台 上 ,以 及 其 他 的 工作 。 当 新 的 代码 被 提交 给 Apache 
Group 时 ,该 团体 审核 它 的 具体 内 容 , 进 行 测试 ,如 果 认 为 满意 ,该 代码 就 会 被 集成 到 
Apache 的 主要 发 行 版 中 。 

Apache 的 优点 : 

(1) 几乎 可 以 运行 在 所 有 的 计算 机 平台 上 。 

(2) 支持 最 新 的 http/1. 1 协议 。 

(3) 简单 而 且 强 有 力 的 基于 文件 的 配置 (httpd. conf) 。 

(4) 支持 通用 网 关 接 口 (cgi) 。 

(5) 支持 虚拟 主机 。 

(6) 支持 Http 认证 。 

(7) 集成 Perl。 

(8) 集成 的 代理 服务 器 。 

(9) 可 以 通过 Web 浏览 器 监视 服务 器 的 状态 , 可 以 自 定义 日 志 。 

(10) 支持 服务 器 端 包含 命令 (ssi) 。 

(11) 支持 安全 Socket 层 (ssl) 。 

(12) 具有 用 户 会 话 过 程 的 跟踪 能 力 。 

(13) 支持 fastcgi。 

(14) 支持 Java Servlets。 

Apache 的 缺点 : 

Apache 没有 为 管理 员 提 供 图 形 用 户 接口 (gui) .但 最 近 的 Apache 版 本 已 经 有 了 gui 
的 支持 。 

2. Apache 的 市 场 情况 

Netcraft 在 Web 服务 器 上 的 统计 (http://www. netcraft. com/survey) 显 示 , 自 从 
1996 年 4 月 以 后 ,Apache 就 成 为 Web 服务 器 领域 应 用 最 为 广泛 的 软件 。 而 在 此 前 ,使 用 
最 广泛 的 Web 服务 器 是 NCSA Web 服务 器 (这 是 Apache 的 前 身 ,也 是 OSS/FS)。 它 在 
1995 年 8 月 至 1996 年 3 月 间 占 据 了 Web 服务 器 市 场 份额 第 一 的 位 置 。 从 2000 年 开始 ， 
Netcraft 就 尝试 只 计算 那些 “活跃 "的 Web 站 点 。 因 为 很 多 Web 站 点 被 创建 以 后 并 未 被 
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使 用 (比如 ,虽然 注册 了 域名 但 并 未 使 用 ) ,这 样 的 站 点 就 属于 * 非 活跃 ”的 站 点 ,很 显然 ,这 
种 统计 方式 更 能 反映 实际 的 情况 。 在 统计 活跃 的 站 点 时 ,2014 年 2 月 的 数据 是 Apache 
占据 了 41. 64% 的 市 场 份额 ,IIS 仅 占据 了 29. 42%, 虽 然 这 几 年 Apache 的 市 场 份 额 有 所 
下 降 , 但 不 容 置疑 的 是 , 它 依然 是 最 主流 的 Web 服务 器 。 如 图 1-6 所 示 , 反 映 的 是 1995 
年 9 月 至 2014 年 2 月 Web 服务 器 市 场 份额 的 变化 情况 (链接 http://news. netcraft. 
com/archives/2014/02/03/february-2014-web-server-survey. html) 。 


Web 服 务 器 发 展 (基于 所 有 网 站 的 市 场 份额 进行 统计 ) 


80% FT 国 Apache 
ETCRA 人 
国 Sun 
国 nginx 
国 Google 
国 NCSA 
国 Other 


o% 一 
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1-6 ”Netcraft 公司 统计 的 Web 服务 器 使 用 情况 


1.1.4 MySQL 简介 


1. MySQL 

MySQL 是 一 个 关系 型 数据 库 管 理 系统 。MySQL 设计 之 初 的 目标 便 是 成 为 高 性 能 、 
普通 用 户 支 付 得 起 的 数据 库 服务 器 和 工具 。 到 今天 为 止 ,这 一 目标 已 经 变 得 越 来 越 近 。 

MySQL 也 代表 开源 精神 。 由 MySQL 带动 的 一 系列 软件 开发 和 经 济 效益 给 数据 库 
领域 带 来 了 许多 革命 性 的 贡献 。MySQL 的 使 命 是 为 所 有 人 贡献 一 个 经 济 实惠 并 且 性 能 
卓越 的 数据 库 软 件 。 

MySQL 的 独到 之 处 一 一 插件 式 存储 引擎 (Pluggable Storage Engine) ,可谓 将 数据 
库 理论 发 挥 得 淋 沉 尽 致 ,完美 地 映射 了 数据 库 的 外 模式 和 内 模式 理论 。MySQL 存储 类 
别 的 粒度 小 到 每 个 表 , 用 户 可 以 最 大 程度 地 利用 各 种 引擎 的 优点 ,又 避免 了 它 的 缺点 。 

相对 于 其 他 数据 库 系 统 往往 无 法 选择 存储 引擎 的 现实 ,MySQL 能 够 满足 多 方面 的 
需求 。 如 数据 库 仓库 市 场 . 典 和 人 式 数据 库 等 。 

2. MySQL 的 市 场 情况 

短 短 的 十 几 年 间 ,MySQL 快速 发 展 ,原本 门 可 罗 淮 的 MySQL 网 站 ,如 今日 访问 超过 
1000 万 页 面 访问 量 (Page View.PV)。 到 目前 为 止 MySQL 装机 量 接近 1200 万 ,而 且 每 
天 下 载 量 以 超过 5 万 的 数量 增加 。 毋 庸 置疑 地 成 为 世界 上 最 流行 的 开源 数据 库 。 

据 MySQL 官方 称 ,MySQL 数据 库 占 有 全 球 数据 库 25% 的 市 场 份额 互联 网 公司 
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80% 的 市 场 份 额 。 特 别 近 几 年 Web 2.0 的 兴起 ,更 是 引爆 了 MySQL 的 应 用 。 国 内 各 大 
互联 网 公司 也 大 量 地 在 其 关键 产品 中 使 用 MySQL ,使 得 国内 MySQL 技术 人 员 的 就 业 形 
势 一 路 走红 。 
3. MySQL 与 其 他 商用 数据 比较 
从 技术 角度 来 说 ,MySQL 也 不 逊色 于 当前 的 其 他 商用 数据 库 。 表 1-2 是 MySQL 5.1 
与 其 他 主流 数据 库 系 统 的 功能 比较 。 
表 1-2 ”MySQL 5.1 与 其 他 主流 数据 库 系统 的 比较 


数据 库 系统 /功能 Oracle MySQL SQL Server DB2 
是 否 开源 /收费 闭 源 / 收 费 开源 /免费 闭 源 /收费 闭 源 /收费 
视图 支持 支持 支持 支持 
触发 器 支持 支持 支持 支持 
存储 过 程 支持 支持 支持 支持 
索引 强 弱 中 中 
事务 支持 强 一 般 中 强 
复杂 查询 强 弱 中 中 
安全 性 强 中 中 强 
快照 支持 不 支持 支持 支持 
易学 性 难 简单 简单 难 
用 户 友好 度 中 一 般 强 中 


目前 ,MySQL 也 有 商业 支持 版 的 收费 产品 。 

4. MySQL 的 发 展 

MySQL AB 是 一 家 公司 。 瑞 典 主 流 媒体 (瑞典 日 报 ) 是 如 此 描述 这 家 公司 的 : 
“MySQL AB 是 开源 软件 数据 库 软 件 MySQL 的 后 台 支 持 公 司 "。MySQL 公司 由 两 个 瑞 
由 人 Michael Widenius、Allan Larsson 和 一 个 芬兰 人 David Axmark 于 1995 年 在 瑞典 注 
册 成 立 。 

2008 年 2 月 .当时 的 业界 开源 老大 Sun Microsystems 动用 10 亿美 元 收购 了 
MySQL ,造就 了 开源 软件 的 收购 最 高 价 。 这 次 交易 给 开源 交易 设立 了 一 个 新 的 基准 。 
MySQL 被 收购 之 后 ,MySQL 图 标 停止 使 用 ,取而代之 的 是 San/ MySQL 图 标 。 

MySQL 和 Sun 合并 之 后 ,推出 了 MySQL 5. 1GA 版 和 MySQL 5.4 Beta 版 。5. 4 的 
推出 照搬 了 4.1 和 5.0 当时 的 开发 模式 ,让 5.4 版 和 6.0 版 并 行 处 于 Beta 开发 阶段 。 

2009 年 ,数据 库 老 大 Oracle 大 笔 一 挥 , 开 出 74 亿美 元 的 支票 ,将 Sun Microsystems 
和 MySQL 通盘 收 于 旗下 。 

作为 “陪嫁 ”给 Oracle 的 重要 数据 库 技术 , MySQL 已 成 为 Oracle 重点 发 展 的 方向 ， 
也 是 Oracle 抗衡 微软 SQL Server 的 法 宝 。 无 论 MySQL 的 未 来 发 展 如何 , 其 数据 库存 储 
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引擎 的 实现 都 是 解决 数据 存储 问题 的 关键 ,掌握 其 存储 机 制 , 即 可 举一反三 ,从 容 应 对 其 
他 任何 一 种 数据 库 产品 。 


1.2 动态 Web 站 点 


1.2.1 B/S 结构 原理 


在 开始 Web 动态 程序 的 开发 之 前 ,必须 先 要 理解 B/S 结构 。 整 个 Web 可 以 分 为 两 
个 重要 的 组 成 部 分 : 客户 端 和 服务 器 端 。 当 客户 端 需要 请 求 特定 的 URL 时 , 它 将 与 服务 
器 建立 连接 ,并 通过 HTTP 协议 发 送 请 求 至 Web 服务 器 。Web 服务 器 接收 到 客户 端的 
请 求 之 后 将 生成 响应 内 容 , 同 时 将 该 响应 内 容 通 过 连接 返回 给 客户 端 。 

正常 情况 下 ,一 个 HTTP 服务 器 会 等 待 浏览 器 发 送 的 请 求 , 并 根据 HTTP 协议 进行 
响应 。 客 户 端 总 是 请 求 某 个 特定 的 文档 。 服 务 器 将 检查 该 请 求 , 同 时 将 客户 端 请 求 的 文 
档 映 射 到 服务 器 本 地 的 文件 系统 中 ,或 者 将 该 请 求 转发 给 一 个 特定 的 应 用 程序 ,并 由 该 应 
用 程序 负责 对 请 求 进行 处 理 , 生 成 响应 内 容 。 一 旦 处 理 完毕 ,服务 器 将 把 处 理 结果 返回 给 
客户 端 ,如 图 1-7 所 示 。 


2? | # |-[ # | 
9 9 


| 浏览 器 


Web 浏 览 器 


HTTP 服 务 器 


图 1-7 B/S 结构 原理 


图 1-7 的 左 侧 部 分 演示 了 一 个 非常 简单 的 系统 结构 : 用 户 与 浏览 器 打交道 ,浏览 器 
接受 用 户 的 输入 和 数据 ,然后 将 请 求 发 送 到 HTTP 请 求 服务 器 ,而 请 求 服务 器 则 从 磁盘 
上 读 取 客户 端 需要 的 文件 ,再 发 送 给 客户 端 。 

1-7 的 右 侧 部 分 描述 了 系统 中 发 生 的 内 容 : 用 户 在 浏览 器 中 输入 URL 或 点 击 某 
个 超 链 接 , 浏 览 器 从 输入 的 URL 或 超 链 接 中 获取 服务 器 地 址 ,然后 与 服务 器 建立 TCP/ 
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IP 连接 。 使 用 该 连接 ,浏览 器 可 以 向 HTTP 服务 器 发 送 一 个 GET 请 求 , 服 务 器 接收 到 
该 URL 地 址 后 从 URL 中 提取 出 请 求 的 资源 名 称 。 

HTTP 服务 器 读 取 请 求 并 对 请 求 进行 处 理 ,使 用 打开 的 TCP/IP 连接 ,服务 器 在 它 
的 响应 中 发 送 客户 端 需要 的 资源 ,发 送 完 毕 后 服务 器 将 关闭 该 连接 。 浏 览 器 接收 到 服务 
器 返回 的 数据 后 , 它 将 检测 这 些 数 据 。HTML 文档 中 可 以 包含 资源 链接 (比如 图 片 、 
Flash 或 Java applets)。 对 于 这 些 资源 ,客户 端 必须 再 次 向 服务 器 发 送 请 求 获取 。 一 旦 所 
有 的 内 容 都 下 载 完 毕 , 浏 览 器 就 可 以 将 这 个 HTML 展现 在 浏览 器 中 。 

PHP 的 工作 原理 也 是 如 此 : PHP 应 用 程序 通过 请 求 的 URL、 所 有 表单 数据 和 已 捕 
获 的 任意 会 话 信息 从 客户 端 获得 信息 ,从 而 确定 应 该 执行 什么 操作 。 如 有 必要 ,服务 器 会 
从 MySQL 数据 库 获得 信息 ,将 这 些 信 息 与 一 些 HTML 模板 组 合 在 一 起 ,并 将 结果 返回 
给 客户 机 。 当 用 户 在 浏览 器 中 导航 时 ,这 个 过 程 重复 进行 。 当 多 个 用 户 访问 系统 时 ,这 个 
过 程 会 并 发 进行 。 但 是 ,数据 流 不 是 单 向 的 ,因为 可 以 用 来 自用 户 的 信息 更 新 数据 库 , 包 
括 会 话 数据 ,统计 数据 和 用 户 提交 的 内 容 ( 比 如 评论 或 站 点 更 新 )。 除 了 动态 元 素 之 外 ,还 
有 静态 元 素 ,比如 图 像 JavaScript 代码 和 层 全 样式 表 (CSS) 。 


1.2.2 Linux 环境 下 的 安装 与 配置 


在 前 面 的 章节 中 已 经 介绍 了 PHP 的 兼容 性 和 它 对 多 平台 支持 。 然 而 对 于 PHP 来 
说 ,应 用 最 为 广泛 以 及 性 能 最 佳 的 软件 组 合 是 在 Linux 平台 上 使 用 Apache 和 MySQL。 

目前 ,几乎 在 所 有 的 Linux 发 布 版 中 都 默认 包含 了 这 些 产品 。Linux 操作 系统 、 
Apache 服务 器 .MySQL 数据 库 以 及 Perl`.PHP 或 者 Python 语言 ,这 些 产品 共同 组 成 了 
一 个 强大 的 Web 应 用 程序 平台 。 但 是 ,Linux 发 行 版 中 的 软件 跟 该 软件 的 最 新 版 相 比 ， 
都 有 一 定 的 滞后 。 所 以 ,也 可 以 选择 自行 安装 或 编译 各 软件 的 最 新 版 本 。 

以 CentOS 为 例 , 如 果 最 初 系统 安装 时 没有 执行 定制 安装 或 完全 安装 ,就 需要 单独 安 
装 PHP 环境 。 可 以 用 以 下 三 种 方法 之 一 来 安装 PHP: 

。 使 用 Red Hat 的 包 管理 器 (RPM)。 

。 从 源 文件 编译 安装 PHP。 

。 使 用 Shell 前 端 软件 包 管理 器 yum 自动 下 载 安装 。 

其 中 使 用 RPM 或 yum 安装 PHP 是 相对 较 容 易 与 方便 的 方法 。 对 于 没有 太 多 经 验 
的 用 户 来 说 ,通常 推荐 使 用 这 两 种 方法 。 所 需要 的 rpm 软件 包 可 以 在 系统 镜像 CD 中 找 
到 ,也 可 以 从 http://www. rpmfind. com 网 站 上 下 载 最 新 的 . rpm 软件 包 。 

而 使 用 从 源 文件 编译 安装 PHP 时 .可 能 还 需要 编译 安装 以 下 软件 包 。 
autoconf: 2. 13 十 (PHP = 5. 4.0) ,2. 59 十 (PHP >=5. 4.0) 。 
automake: 1. 4 十 。 
libtool: 1. 4. x 十 (除了 1.4.2)。 
re2c: 版 本 0. 13. 4 或 更 高 。 
flex: 版 本 2. 5.4(PHP 二 =5. 2)。 
bison: 版 本 1. 28( 建 议 ),1. 35 或 1.75。 
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1.2.3 Windows 环境 下 的 安装 配置 


在 Windows 平台 安装 PHP 时 ,可 以 从 很 多 选项 中 进行 选择 ,如 与 Windows 一 起 使 
用 Apache 和 MySQL ,或 者 使 用 Microsoft SQL Server 和 IIS 的 组 合 。 

1. 下 载 软件 

1) MySQL 

从 http://dev. mysql. com/downloads/ 上 下 载 MySQL 数据 库 服务 器 推荐 的 通用 
(MySQL Community Server) 版 本 。 

MySQL 将 指定 推荐 版 本 ,它们 将 是 最 新 发 布 的 稳定 版 本 。 在 编写 本 书 时 ,最 新 版 本 
是 5.6.16。 在 下 载 页 面 上 ,应 该 下 载 针 对 Windows 系统 的 版 本 。 

2) Apache 

从 http://httpd. apache. org/download. cgi 上 下 载 Apache 的 最 新 版 本 。 

Apache 的 版 本 有 三 个 分 支 : 1. 3. x、2. 0. x 和 2. 2. x。 在 编写 本 书 时 ,使 用 的 是 
Apache 2. 2 版 本 的 最 新 版 本 2.4.7。 

Apache for Windows 可 以 作为 一 个 简单 的 可 执行 文件 使 用 。 

3) PHP 

从 http://windows. php. net/download/ 下 载 PHP for Windows 的 最 新 版 本 。 

下 载 Windows Zip 程序 包 。 在 编写 本 书 时 ,最 新 的 版 本 是 5. 5. 9。 

2. 安装 软件 

1) MySQL 

从 MySQL 的 Web 站 点 上 下 载 的 文件 将 是 MSI 类 型 , 它 是 一 种 常见 的 Windows 安 
装 程序 。 直 接 双击 下 载 的 文件 ,开始 安装 过 程 。 

安装 过 程 可 根据 程序 向 导 进 行 ,值得 注意 的 是 ,安装 完成 后 开始 配置 MySQL ,全 部 
保持 默认 选项 即 可 ,但 最 好 把 MySQL 默认 编码 改 为 utf8, 如 图 1-8 所 示 。 


MYSQL! Server Instance Configuration 
Configure the MySQL Server 5.5 Server instance. 


Please select the default character set. 


Standard Character Set 
Makes Latinl the default charset, This character set is suited for 
English and other West European languages. 


CF Best Support For Multiingualism 
Make UTFS the default character set. This is the recommended 
character set for storing text in many different languages. 


® Manual Selected Default Character Set / Collation 
Please specity the character setto use. 


charaderset ”|[ 面 了 
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需要 在 安装 过 程 中 设 定 root 用 户 的 密码 。 根 用 户 root 具有 对 MySQL 的 不 受 限制 
的 访问 权限 ,因此 这 个 密码 应 该 是 安全 的 ,并 且 不 应 该 忘记 它 。 在 Modify Security 
Settings 选项 中 设置 密码 ,输入 两 次 密码 即 可 完成 ,最 后 单 击 Next 按钮 完成 配置 ,如 图 1-9 
所 示 。 


Configure the MySQL Server 5.5 server instance. 


Please set the security options. 
[7 Modify Security Settings 


蜂 New root password: 。「… Enter the root password. 
Confim 


ed Retype the password. 


厂 Enable root access from remote machines 


厂 Greate An Anonymous Account 


This option will create an anonymous account on this server, 
Please note that this can lead to an insecure system. 


图 1-9 设置 root 账户 密码 


2) Apache 

从 Apache 的 Web 站 点 上 下 载 的 文件 也 是 MSI 类 型 ,双击 下 载 的 安装 程序 ,开始 安 
装 过 程 ,逐步 通过 向 导 。 

安装 程序 将 经 过 多 个 步骤 ,包括 同意 许可 。 当 要 求 输入 服务 器 信息 时 ,可 以 输入 几乎 
所 有 的 信息 。 尽 管 作为 一 种 服务 应 该 选择 为 所 有 用 户 运 行 Apache。 在 安装 过 程 中 ,还 需 
选择 目标 文件 夹 ,默认 是 C:\Program Files\Apache Group, 最 后 完成 实际 的 安装 过 程 。 

安装 完成 之 后 ,在 浏览 器 输入 http://localhost/ ,如果 显示 “It Works!”, 则 表示 
Apache 安装 成 功 ,如 图 1-10 所 示 。 


Oe acxje :ER 


It works! 


图 1-10 Apache 安装 成 功 测试 页 面 


3) PHP 

PHP 的 安装 过 程 稍微 复杂 一 点 。 

(1) 需要 解压 缩 下 载 的 文件 。 

可 以 把 它 的 内 容 放 到 几乎 任何 位 置 ,但 是 选择 C:\php 有 意义 一 些 。 不 要 在 目录 路 
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径 中 使 用 空格 ,因为 这 可 能 导致 某 些 服务 器 崩溃 。 

(2) 把 php. ini-development 复制 到 Windows 的 目录 中 ,并 将 其 重 命名 为 php. ini。 

php. ini 文件 控制 PHP 的 行为 方式 。Zip 程序 包 带 有 这 个 文件 的 两 个 示例 ,php. ini- 
development 是 其 中 的 一 个 文件 。 应 该 把 它 复制 或 移动 到 Windows 目录 中 ,可 能 是 C:\ 
WINNT 或 C:\Windows, 或 者 类 似 的 目录 ,这 取决 于 特定 的 操作 系统 。 把 这 个 文件 重 命 
名 为 php. ini。 

(3) 从 “开始 ”菜单 中 选择 “程序 ”>Apache HTTP Server~Configure Apache Server 
一 Edit the Apache httpd. conf Configuration File 命令 。 

现在 ,需要 告诉 Apache 把 PHP 用 于 某 些 文件 。 为 了 执行 该 任务 ,需要 编辑 Apache 
的 配置 文件 (httpd. conf) 。 幸 运 的 是 ,Apache 安装 程序 为 此 创建 了 一 个 “开始 ”菜单 快捷 
方式 ,应 该 会 在 “记事 本 "或 男 一 个 文本 编辑 器 中 打开 文件 。 

(4) 在 LoadModule 部 分 的 末尾 ,添加 : 

LoadModule php5 module "c:/php/php5apache2 2.d11" 


通过 手动 编辑 Apache 的 配置 文件 ,以 启用 PHP。 
这 一 步 告诉 Apache 加 载 PHP 模块 。 

(5) 添加 index. php 作为 目录 索引 。 

找 出 以 DirectoryIndex 开头 的 行 ,取消 这 一 行 的 注释 : 


DirectoryIndex index.html index.php 


这 允许 index. php 作为 目录 中 的 主 文件 ,例如 ,如 果 用 户 只 输入 了 www. sitename 
.com, 则 Apache 将 提供 www. sitename. com/index. php。 

(6) 重新 启动 Apache, 使 以 上 配置 生效 。 

(7) 测试 PHP。 

最 后 ,可 以 通过 运行 脚本 phpinfo. php ,测试 是 否 启用 了 PHP, 并 会 显示 关于 PHP 安 
装 的 大 量 信息 。 这 个 脚本 非常 简单 ,但 它 是 PHP 开发 人 员 曾 经 编写 过 的 最 重要 的 脚本 
之 一 ,因为 它 提供 了 非常 多 的 有 价值 的 知识 。 

把 phpinfo. php 文件 放置 在 服务 器 上 正确 的 目录 中 ,这 依赖 于 操作 系统 和 Web 服务 
器 。 对 于 安装 了 Apache 的 Windows 用 户 ,这 个 目录 名 为 htdocs, 并 且 在 Apache 目录 ， 
默认 为 C:\Program Files\Apache Group\Apache 内 。 

如 果 在 Web 浏览 器 中 运行 PHP 脚本 时 , 它 试 图 下 载 文件 ,那么 Web 服务 器 就 不 会 
把 文件 扩展 名 识别 为 PHP。 可 以 检查 Apache 的 配置 ,以 校正 这 个 问题 。 


1.2.4 常见 Apache 十 PHP 十 MySQL 整合 安装 环境 


虽然 可 以 遵照 这 些 指导 并 手动 安装 所 有 必需 的 软件 ,但 也 有 另 一 种 选择 。 可 以 在 线 
找到 多 种 不 同 的 免费 多 合 一 程序 包 , 它 们 可 以 一 次 性 安装 Apache、PHP 和 MySQL 一 一 
有 时 还 包括 像 phpMyAdmin 这 样 的 额外 软件 。 其 缺点 是 : 如 果 安 装 发 生 问题 , 则 可 能 更 
难以 调试 。 
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1. AppServ (推荐 ,简洁 精简 ) 

主页 ; http://www. appservnetwork. com/。 

当前 的 两 个 版 本 是 2. 5. 10 及 2. 6. 0。 

AppServ 是 PHP 网 页 架 站 工具 组 合 包 ,作者 将 一 些 网 路 上 免费 的 架 站 资源 重新 包装 
成 单一 的 安装 程序 ,以 方便 初学 者 快速 完成 架 站 , AppServ 所 包含 的 软件 有 Apache、 
Apache Monitor.PHP、MySQL、PHP-Nuke、phpMyAdmin。 适 合 初学 者 

2. XAMPP (功能 全 面 ) 

主页 http://www. apachefriends. org/zh_cn/index. html。 

XAMPP 是 一 款 具有 中 文 说 明 的 功能 全 面 的 集成 环境 , XAMPP 并 不 仅仅 针对 
Windows, 而 是 一 个 适用 于 Linux、Windows、Mac OS X 和 Solaris 的 易于 安装 的 Apache 
发 行 版 。 软 件 包 中 包含 Apache 服务 器 、MySQL、 SQLite、PHP、 Perl, FileZilla FTP 
Server、Tomcat 等 。 默 认 安装 开放 了 所 有 功能 ,安全 性 有 问题 ,需要 对 以 下 安全 问题 进行 
设 定 : 

。 MySQL 管理 员 (root) 未 设置 密码 。 

。 MySQL 服务 器 可 以 通过 网 络 访问 。 

PhpMyAdmin 可 以 通过 网 络 访问 。 
。 样 例 可 以 通过 网 络 访问 。 
Mercury 邮件 服务 器 和 FileZilla FTP 服务 器 的 用 户 是 公开 的 。 

3. WampServer (简便 易 用 ) 

主页 http://www. wampserver. com/en/。 

WampServe 集成 了 Apache、MySQL、PHP、phpmyadmin, 支持 Apache 的 mod_ 
rewrite,PHP 扩展 .Apache 模块 有 相应 的 “开启 /关闭 "菜单 方便 管理 ,省 去 了 修改 配置 文 
件 的 麻烦 。 

4. phpstudy 

个 人 主页 http://www. phpstudy. net/ 。 

该 程序 包 集成 最 新 的 Apache 十 PHP 十 MySQL 十 phpMyAdmin 十 ZendOptimizer ,一 
次 性 安装 ,无 须 配 置 即 可 使 用 ,是 非常 方便 .好 用 的 PHP 调试 环境 。 该 程序 不 仅 包 括 
PHP 调试 环境 ,还 包括 了 开发 工具 、 开 发 手册 等 。 总 之 学 习 PHP 只 需 一 个 包 。 

对 于 学 习 PHP 的 新 手 来 说 ,不 管 是 Windows 还 是 Linux, 全 新 且 完 整 的 环境 配置 是 
一 件 很 困难 的 事 , 对 于 有 经 验 的 开发 者 来 说 也 是 一 件 烦琐 的 事 。 因 此 无 论 是 新 手 还 是 老 
手 , 以 上 列 出 的 程序 包 套 件 都 是 一 个 不 错 的 选择 。 


本 章 小 结 


本 章 重点 讲述 了 什么 是 PHP、PHP 的 语言 优势 及 发 展 , 介 绍 了 开源 的 历史 背景 ,以 
及 与 PHP 配合 使 用 的 Apache 和 MySQL 软件 介绍 ,通过 这 些 内 容 使 读者 对 PHP 有 一 
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个 全 面 的 认识 。 
重点 回顾 
1. PHP、Apache 及 MySQL 各 软件 的 功能 。 
2. 开源 的 意义 。 
3. AMP 环境 在 不 同系 统 中 的 安装 过 程 。 
4. 整合 套件 的 用 法 。 
本 章 实 训 
【 实 训 】 


选择 一 种 合适 的 安装 方式 ,为 后 续 的 学 习 搭 建 开发 环境 。 
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PHP 是 一 种 易于 学 习 和 使 用 的 服务 器 端 脚本 语言 ,只 需要 很 少 的 编程 知识 就 能 使 用 
PHP 建立 一 个 真正 交互 的 Web 站 点 。 初 学 者 需要 具备 HTML、JavaScript 等 脚本 语言 
基础 ,这 些 知 识 为 学 习 PHP 打下 了 坚实 的 基础 。 

从 语法 上 看 ,PHP 语言 近似 于 C 语言 。 可 以 说 ,PHP 是 借鉴 C 语言 的 语法 特征 ,由 
C 语言 改进 而 来 的 。 


2.1 基本 语法 


2.1.1 在 Web 页 面 中 嵌入 PHP 


虽然 可 以 书写 和 运行 独立 的 PHP 程序 ,但 是 大 多 数 的 PHP 代码 都 是 嵌入 到 HTML 
文件 中 的 。 这 上 毕竟 是 PHP 被 创造 的 首要 原因 。PHP 是 一 种 HTML 中 艇 入 的 脚本 语言 。 
HTML 中 嵌入 的 (HTML-embedded) 是 指 可 以 把 PHP 代码 和 HTML 代码 混合 在 相同 
的 脚本 里 ,不仅 可 以 将 PHP 脚本 嵌入 到 HTML 文件 中 ,还 可 以 把 HTML 标签 也 嵌入 在 
PHP 脚本 里 。 

1. PHP 标签 的 类 别 

为 了 开始 用 PHP 编程 .首先 从 一 个 简单 的 Web 页 面 开 始 。 脚 本 2-1 是 一 个 简洁 的 
XHTML 过 渡 性 文档 的 示例 , 它 将 用 作 本 书 中 每 个 Web 页 面 的 基础 。 


脚本 2-1 基本 XHTML 1.0 过 渡 型 Web 文档。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 


<title> 网 页 标题 < /title> 
</head> 
<body> 
</body> 
10 </htm> 


1 
3 
4 
5 <meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
6 
8 
9 
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因为 一 个 单独 的 文件 中 包含 PHP 和 非 PHP 的 源 代码 ,所 以 需要 一 种 方法 来 识别 出 
属于 PHP 代码 的 区 域 用 于 执行 。PHP 提供 了 多 种 不 同 的 方式 。 
(1) 首选 的 正式 语法 。 


<?Php 


bk 

(2) 非 正式 的 语法 。 

区 多 

区 六 

这 些 符 号 叫做 PHP 标签 (PHP Tags) ,是 用 来 告诉 Web 服务 器 PHP 程序 从 哪里 开 
始 、 到 哪里 结束 。 标 签 内 的 所 有 内 容 都 会 被 当 作 PHP 来 解释 ,这 意味 着 PHP 解释 器 将 
处 理 这 些 代码 ;PHP 标签 外 的 文本 会 被 当 作 一 般 的 HTML, 立 即 发 送 给 Web 浏览 器 。 
PHP 标签 让 我 们 可 以 脱离 HTML。 

注 : 当 使 用 “二 ?…? 二 ”将 PHP 代码 嵌入 HTML 文件 中 时 ,可 能 会 同 XML 发 生 冲 
突 ,同时 ,能 和 否 使 用 这 一 缩减 形式 还 取决 于 PHP 本 身 的 设置 。 

对 PHP 脚本 的 最 后 一 个 考虑 是 文件 必须 使 用 正确 的 扩展 名 。 扩 展 名 告诉 服务 器 把 
脚本 作为 PHP 页 面 处理 。 大 多 数 Web 服务 器 都 为 标准 HTML 页 面 使 用 . html 或 . htm 
扩展 名 ,通常 为 PHP 脚本 首选 . php 扩展 名 。 

2. 建立 基本 的 PHP 脚本 

(1) 在 文本 编辑 器 中 创建 一 个 新 文档 ( 见 脚本 2-2) 。 


脚本 2-2 这 是 第 一 个 PHP 脚本 , 它 本 身 不 做 任何 事情 ,只 示范 了 要 使 用 的 语法 。 


1 <!DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll1- transitional.dtd"> 

3 <html xmlns="http://www.w3.0rg/1999/xhtml"> 

4 <head> 

5 <meta http-equiv= "Content- Type" content="text/html; charset=gb2312" /> 
6 ”<title> 最 简单 的 PHP 页 面 < /title> 

7 </head> 

8 <body> 

9 <p> 这 里 是 HIML。< /p> 

10 <?php 

11 ?> 

12 </body> 

13 </html> 


一 般 可 以 任意 选择 要 使 用 的 文本 编辑 器 , 它 可 以 是 Macintosh 上 的 BBEdit、 
Windows 上 非常 基本 的 “记事 本 ”或 更 高 级 的 Dreamweaver, 或 者 是 Linux 上 的 vi。 

(2) 开始 编写 HTML 文档 。 

强烈 建议 PHP 开发 人 员 使 用 正式 的 PHP 标签。 
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(3) 将 文件 另存 为 first. php。 

记 住 : 如 果 没 有 使 用 合适 的 PHP 扩展 名 保存 文件 ,该 脚本 将 不 会 正确 执行 。 

(4) 将 该 文件 置 于 Web 服务 器 上 正确 的 目录 中 。 

如 果 在 自己 的 计算 机 上 运行 PHP, 只 需要 将 文件 保存 到 计算 机 上 的 特定 文件 夹 中 。 
可 以 检查 特定 Web 服务 器 应 用 程序 的 文档 ,以 找到 该 目录 。 常 见 的 选项 有 C:\inetpub\ 
wwwroot(Windows 上 的 IIS)、C:\Program Files\ Apache Group\ Apache \ htdocs 
(Windows 上 的 Apache) ,或 者 一 /Sites( 具 有 Apache 的 Mac OS X, 其 中 一 指 主 目录 ) ,或 
者 /var/www/html(Linux 上 的 Apache) 。 

如 果 在 远程 托管 服务 器 上 运行 PHP, 则 需要 使 用 FTP 应 用 程序 将 文件 上 传 到 正确 
的 目录 中 ,托管 ISP 运营 商 将 提供 访问 权限 及 其 他 必要 的 信息 。 

(5) 在 Web 浏览 器 中 运行 first. php, 如 图 2-1 所 示 。 


Er 
S 


图 2-1 first. php 在 浏览 器 中 的 运行 结果 


图 2-1 虽然 看 起 来 与 任何 其 他 简单 的 HTML 页 面 一 样 ,但 这 是 一 个 PHP 脚本 , 它 是 
本 书 中 其 余 示 例 的 基础 。 

同样 ,如 果 在 自己 的 计算 机 上 运行 PHP, 将 需要 浏览 地 址 http://localhost/first. php 
或 http://localhosUV 一 二 username 二 /first. php。 如 果 正 在 使 用 一 台 具 有 个 人 域名 的 
Web 主机 , 则 需要 使 用 http://your-domain-name /first. php。 

正常 情况 下 ,应 该 看 到 一 个 简单 但 完全 有 效 的 Web 页 面 。 如 果 看 到 了 PHP 代码 ( 标 
签 ), 则 说 明 没 有 使 用 正确 的 扩展 名 ,或 者 服务 器 不 支持 PHP, 需 要 检查 文件 及 服务 器 环 
境 配置 ,排除 相关 错误 。 

由 于 PHP 脚本 需要 由 服务 器 解析 ,所 以 必须 通过 URL(http://localhost/first. php 
或 http://www. dmcinsights. com/first. php) 访 问 PHP 脚本 。 不 能 像 在 HTML 静态 页 
面 或 其 他 应 用 程序 中 打开 一 个 文件 那样 在 Web 浏览 器 中 简单 地 打开 它们 ,如 果 这 样 做 ， 
则 地 址 的 开头 部 分 将 是 file://。 

另外 ,可 以 在 单一 HTML 文档 中 嵌入 PHP 代码 的 多 个 部 分 , 即 可 以 在 两 种 语言 
间 来 回转 换 。 


2.1.2 发 送 数据 到 Web 浏览 器 


要 利用 PHP 构建 动态 Web 站 点 ,必须 知道 如 何 发 送 数据 到 Web 浏览 器 。PHP 具 
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有 许多 用 于 此 目的 的 内 置 函 数 , 其 中 最 常用 的 是 echoO) 和 print()。PHP 和 HTML 最 简 
单 的 交互 是 通过 print 和 echo 语句 来 实现 的 ,在 实际 使 用 中 ,print 和 echo 两 者 的 功能 几 
乎 是 完全 一 样 。 可 以 这 么 说 , 凡 一 个 可 以 使 用 的 地 方 , 另 一 个 也 可 以 使 用 。 


echo 'Hello, world!'; 

print "您 好 !"; 

正如 从 这 个 示例 中 可 以 看 到 的 , 任 一 个 函数 都 将 使 用 单 引 号 或 双 引 号 。 应 注意 ; 在 
PHP 中 ,所 有 语句 都 必须 以 分 号 结尾 。 


1. 


将 数据 输出 到 Web 浏览 器 


(1) 在 文本 编辑 器 中 打开 first. php( 参 见 脚 本 2-2) 。 
(2) 在 PHP 标签 (第 10 行 和 第 11 行 ) 之 间 ,添加 一 条 简单 的 消息 (参见 脚本 2-3)。 


echo ' 这 一 行 由 PHP 输 出 。'"'; 


脚本 2-3 PHP 可 以 使 用 print() 或 echo() 发 送 数据 到 Web 浏览 器 (如 图 2-2 所 示 )。 


1 
3 
4 
5 
6 
7 
8 
9 


< !DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title> 使 用 echo ()< /title> 
</head> 
<body> 
<p> 这 里 是 HIML。< /p> 
<?php 
echo ' 这 一 行 由 PHP 输 出 。'; 
ke 
< /body> 
</htm> 


在 此 ,对 于 在 这 里 输入 的 消息 内 容 \ 使 用 的 是 哪个 函数 (echo() 或 print()) 或 者 哪 种 
引号 ,都 是 无 关 紧 要 的 。 
(3) 可 以 按 自 己 的 想法 更 改 页 面 标 题 (第 6 行 )。 


<title> 使 用 echo()< /title> 


这 是 可 选 的 ,并 且 纯粹 是 装饰 性 的 改变 。 

(4) 将 文件 另存 为 second. php,. 上 传 到 Web 服务 器 ,然后 在 Web 浏览 器 中 测试 它 
(如 图 2-2 所 示 ) 。 

如 果 看 到 的 是 解析 错误 ,而 不 是 输出 的 消息 ,如 图 2-3 所 示 .请 检查 是 否 同时 具有 开 
引号 和 闭 引 号 ,以 及 对 任何 有 问题 的 字符 转 义 ,还 要 肯定 每 一 条 语句 都 用 分 号 结尾 。 

如 果 看 到 的 是 完全 空白 的 页 面 , 这 可 能 是 由 于 下 面 两 个 原因 之 一 引起 的 : 


PHP+MySQL 项 目 实例 开发 


这 一 行 由 PHP 输 出 。 


图 2-2 ”second. php 在 浏览 器 中 的 运行 结果 


Eee -exe “ 国 


Parse error: syntax error, unexpected T_ECHO, 
expecting ’,” or ’;” in C:\AppServ\www\book\ch02 


\second. php on line 12 


2-3 解析 错误 


(1) HTML 代码 有 问题 。 通 过 查看 页 面 的 源 文件 来 测试 它 , 并 寻找 其 中 的 HTML 
问题 。 

(2) 发 生 了 PHP 的 错误 ,但 是 PHP 配置 中 关闭 了 display_errors, 因 此 ,不 会 显示 任 
何 内 容 。 

2. echo() 和 print() 

从 技术 上 讲 ,echo() 和 print() 是 语言 构造 ,而 不 是 函数 ,所 以 双 括 号 不 是 必需 的 ,如 
脚本 2-3 中 ,第 11 行 代码 “echo ' 这 一 行 由 PHP 输出 。'; ?未 使 用 双 括号 。 

PHP 对 函数 名 不 区 分 大 小 写 , 因 此 ,ECHO()、echo()、eCHo() 等 都 有 效 。 另 外 ， 
echo() 和 print() 还 可 以 用 来 发 送 HTML 代码 到 Web 浏览 器 ,其 方法 如 下 ,效果 如 图 2-4 
所 示 。 


echo '<b>Hello, <font size= "+2"> world< /font> !</b>'; 


. 
| teresa pox] 


这 里 是 HTIL。 


Hello，WOT1d' 


图 2-4 PHP 可 以 发 送 HTML 代码 到 Web 浏览 器 
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但 是 ,两 者 之 间 还 有 一 个 非常 重要 的 区 别 : echo() 可 以 同时 输出 多 个 字符 串 ,发 送 多 
个 单独 的 数据 块 到 Web 浏览 器 ,它们 之 间 用 逗号 分 隔 开 ,而 print() 则 只 可 以 同时 输出 一 
个 字符 串 。 例 如 : 


echo 'Hello,', 'world!'; 


2.1.3 ”理解 PHP、HTML 和 空白 


1. PHP 的 空白 .HTML 的 空白 与 Web 页 面 的 空白 

在 进一步 深入 探讨 使 用 PHP 发 送 HTML 和 其 他 数据 到 Web 浏览 器 之 前 ,充分 理 
解 PHP 的 处 理 过 程 是 重要 的 。 利 用 PHP, 可 以 发 送 数据 CHTML 标签 和 文本 的 组 合 ) 到 
Web 浏览 器 。 接 下 来 , Web 浏览 器 可 以 将 其 显示 为 最 终 用 户 查 看 的 Web 页 面 。 因 此 , 利 
用 PHP 所 做 的 工作 就 是 创建 Web 页 面 的 HTML 源 文件 (HTML source)。 形 象 地 讲 ， 
这 意味 着 PHP 正在 生成 如 图 2-5 所 示 的 HTML( 通 过 在 浏览 器 中 选择 “查看 ”一 “ 源 文 
件 ” 命 令 或 “查看 “页 面 源 文件 ”命令 来 访问 它 ), Web 浏览 器 将 把 它 转变 成 如 图 2-6 所 
示 的 内 容 。 


0 0 L 总 0 50 
1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTNL 1.0 Transitional//EN"” 
2 "http://wwwy.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd”> 

3 <html xmlns="http://wuw.v3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=gh2312" /> 

6 <cicle> 空 白 </cicle> 

7 </head> 

le 

9 <body> 

0 <b>Hello, <font size="+2"> world</font>!</b><br /><pre> 。 缩 进 2 个 字 。</prey> 
1 </body> 

2 </html> 


图 2-5 使 用 PHP 生成 Web 页 面 的 HTML 源 文件 


Hello, World! 


久 进 2 个 字 。 


图 2-6 Web 浏览 器 将 它 转变 成 良好 格式 化 的 显示 


记 住 这 一 点 ,实质 上 可 以 在 3 个 地 方 产生 空白 ,影响 间距 : 在 PHP 脚本 中 ,在 
HTML 源 文件 中 以 及 在 呈现 的 Web 页 面 中 。 创 建 的 额外 的 空格 、 制 表 位 以 及 空白 行 一 
般 被 称 为 空白 (white space) 。 

PHP 一 般 会 忽略 空白 ,这 意味 着 在 编写 代码 时 可 以 加 宽 代码 间距 ,不 管 怎样 ,编程 人 
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员 和 希望 自己 的 脚本 更 易 读 。HTML 一 般 也 会 忽略 空白 。 确 切 地 讲 , HTML 中 会 影响 页 
面 的 唯一 空白 是 单个 空格 ,多 个 空格 仍然 作为 一 个 空格 呈现 。 因 此 ,图 2-7 中 的 HTML 
将 生成 与 图 2-5 中 的 HTML 相同 的 页 面 ,如 图 2-6 所 示 。 


FE. FT ,RP A 

1 <!DOCTYPE html PUBLIC "-//W3c//DTD XHTNL 1.0 Transitional//EN" 

2 "http://waw.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 

3 <html xmlns="http://wwy.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv="Content-Type" content="text/html; charset=gb2312" />| 
6 <cicle> 空 白 </citle> 

7 </head> 


日 
9 <body> 


<b>Hello, <font size="+2"> world</font>!</b> 
<br /> 


缩 进 2 个 字 。</pre> 


图 2-7 HTML 源 文件 中 的 空白 


2. 创建 空白 

。 要 创建 Web 页 面 的 空白 ,改变 完成 的 Web 页 面 的 间距 ,可 以 使 用 HTML 标签 
二 br /二 (换行 标签 ,在 较 旧 的 HTML 标准 中 是 一 br>)、 一 p 二 一 /p 二 (段落 标 
签 ) 和 &nbsp;( 空 格 ) 。 

。 要 创建 HTML 的 空白 ,改变 用 PHP 创建 的 HTML 源 文件 的 间距 ,可 以 在 多 个 行 
上 使 用 echo() 或 print() ;或 者 在 echo 时 双 引 号 内 使 用 换行 符 (\n)。 

(1) 在 文本 编辑 器 中 打开 second. php( 参 见 脚本 2-3) 。 

(2) 如 果 需 要 ,可 更 改 页 面 的 标题 (参见 脚本 2-4) 。 


<title> 空 白 < /title> 


脚本 2-4 这 个 脚本 演示 了 PHP 代码 .HTML 源 文件 和 呈现 的 Web 页 面 内 创建 的 不 
同类 型 的 空白 。 


1 < !DOCTYPE html PUBLIC "- //W3C//DID XHTML 1.0 Transitional//EN" 

2 "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 

3 <html xmlns= "http://www.w3-org/1999/xhtml"> 

4 <head> 

5 <meta http- equiv= "Content- TYpe"” content= "text/html; charset=gb2312" /> 
6 <title> 空 白 </title> 

7 </head> 

8 <body> 

9 <?php 

10 


11 echo ' 这 里 的 echo () 语 名 
12 分 为 两 行 输出 。'; 
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14 echo "<br /> 这 一 行 单独 显示 .\n\n"; 
16 ”echo ' 最 后 一 行 ."'; 


18 ?> 

19 </body> 

20 </html> 

(3) 在 初始 PHP 标签 之 后 , 按 下 Enter 键 。 

Enter 键 创 建 的 空白 行 不 会 影响 PHP 脚本 、HTML 源 文件 或 呈现 的 HTML 页 面 ， 
但 是 它 使 得 PHP 脚本 更 易 读 。 

(4) 修改 现 有 的 echo() 语 句 , 使 得 它 分 布 在 多 行 上 。 

echo ' 这 里 的 echo () 语 句 

分 为 两 行 输出 。'; 

一 旦 执行 这 个 脚本 ,在 echo 语句 中 间 按 下 Enter 键 ,使 得 “分 为 ”出 现在 下 一 行 的 开 
始 处 ,将 会 在 两 行 上 生成 HTML 源 文件 。 这 是 有 效 的 ,因为 echo() 语 句 会 继续 工作 , 直 
至 它 遇 到 结束 的 单 引 号 。 

(5) 输入 另 一 条 echo() 语 句 , 它 使 用 换行 标签 和 换行 符 。 

echo "<br /> 这 一 行 单 独 显示 .\n\n"y 

在 这 条 语句 中 ,正在 完成 两 个 目标 。 首 先 ,发 送 了 一 个 换行 标签 二 br /二 ,使 得 Web 
浏览 器 在 其 自己 的 行 上 显示 这 个 句子 。 其 次 ,用 两 个 换行 符 结 束 了 这 条 语句 ,来 影响 
HTML 源 文件 。 为 了 使 换行 符 工作 ,必须 使 用 双 引 号 。 

(6) 添加 最 后 一 条 echo() 语 句 。 


echo ' 最 后 一 行 ."; 


(7) 将 该 文件 另存 为 whitespace. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 
它 , 如 图 2-8 所 示 。 


人 @ htpyllocalhosybe P - © || Se 


这 里 的 echo 〇 语句 分 为 两 行 输出 。 
这 一 行 单独 显示 .最 后 一 行 . 


2-8 改变 显示 的 Web 页 面 间距 的 唯一 方式 是 使 用 HTML 的 标签 


应 该 看 到 ,该 脚本 在 最 终 的 Web 页 面 中 创建 间距 的 唯一 方法 是 通过 换行 标签 二 br / 
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完成 , 它 在 “输出 ”和 “这 一 行 "之 间 打 印 。 
(8) 查看 页 面 的 HTML 源 文件 ,如 图 2-9 所 示 。 


FPP PRP 0 PTFFE UPPER FPF 60 ， 
1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTNL 1.0 Transitional//EN" 
日 "http://wuy.w3.org/TR/xhtml1/DTD/xhtml1i-transitional.dtd"> 
3 <html xmlna="http://wwu.v3.org/1999/xhtml"> 
4 <head> 
5 <meta http-equiv="Content-Type" content="text/html; charset=gh2312" /> 
6 <title> 空 白 </title> 


a 
3 这 里 区 echo 0 语句 
分 为 两 行 输 出 。<br /> 这 一 行 单独 显示 . 


1 
z 最 后 一 行 .</body> 


图 2-9 PHP 脚本 创建 的 HTML 代码 


在 HTML 源 文件 中 ,可 以 看 到 没有 传输 PHP 代码 中 的 额外 间距 。 把 一 条 语句 分 为 
多 行 (第 11 行 和 第 12 行 ) 或 使 用 换行 符 \n( 参 见 脚本 2-4 的 第 14 行 ) 确 实 会 影响 HTML 
源 文件 ,但 是 不 会 影响 呈现 的 Web 页 面 。 

因此 ,在 PHP 脚本 中 慷慨 地 使 用 空白 间距 ,将 使 它们 更 易于 编码 和 编辑 。 当 使 用 
PHP 生成 HTML 时 ,应 该 尽 可 能 设法 使 HTML 源 文件 最 易 读 ,以 便 细 读 和 调试 原始 
HTML 源 文件 。 


2.1.4 编写 注释 


编写 执行 的 PHP 代码 自身 只 是 编程 过 程 的 一 部 分 。 动 态 Web 站 点 开发 的 一 个 次 要 
但 仍然 至 关 重 要 的 方面 涉及 为 代码 加 注解 。 
在 HTML 中 ,可 以 使 用 以 下 方式 添加 注释 : 


<!--HIML 注 释 --> 


HTML 注释 在 源 文件 中 可 以 看 到 .如 图 2-10 所 示 , 但 是 不 会 出 现在 呈现 的 Web 页 
面 中 。 


1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTNL 1.0 Transitional//EN" 

z "http://weu.v3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
3 <html xmlns="http://www.w3.org/1999/xhtml"> 

4 <head> 


5 <mera http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
5 <ticle> 空 白 </cicle> 

7 </head> 

8 <body> 

9 <!-- PHP 生 成 的 代码 开始 --> 


o 
1 这 里 的 echo() 语 句 
2 分 为 两 行 输 出 。<br /> 这 一 行 单独 显示 . 


4 最 后 一 行 .<!-- pap 生 成 的 代 玛 结束 -> 


6 </body> 
了 </hcml> 


2-10 HTML 注释 
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PHP 注释 则 有 所 不 同 , 因 为 它们 根本 不 会 被 发 送 到 Web 浏览 器 ,这 意味 着 最 终 用 户 
不 会 看 到 它们 ,即使 查看 HTML 源 文件 也 是 如 此 。 


本 


PHP 支持 3 种 注释 类 型 。 


第 一 种 使 用 井 号 (# ) 。 
# 这 是 Unix Shell 风格 的 单行 注释 


第 二 种 使 用 两 个 斜 杠 。 


// 这 是 c++ 风格 的 单行 注释 


这 两 种 注释 都 会 使 PHP 忽略 其 后 直到 这 一 行 末 的 一 切 内 容 。 因 此 ,这 两 种 注释 都 


只 是 单行 注释 。 它 们 还 常用 于 在 与 某 些 PHP 代码 相同 的 行 上 添加 注释 。 


Print 'Hello!'; //Say hello. 


第 


三 种 风格 允许 注释 分 布 在 多 行 上 。 


/* 这 是 C 风 格 的 多 行 注释 
分 为 两 行 。* / 


2. 


给 脚本 加 注释 


(1) 在 文本 编辑 器 中 打开 whitespace. php( 参 见 脚 本 2-4) 。 
(2) 在 初始 PHP 标签 之 后 ,编写 第 一 条 注释 (参见 脚本 2-5) 。 
# 创 建 日 期 :2014- 11- 11 


# 创 建 者 : 张 三 
# 本 脚本 用 于 测试 PHP 注释 


脚本 2-5 这些 基本 的 注释 演示 了 在 PHP 中 可 以 使 用 的 3 种 语法 。 


1 
2 
3 
4 
5 
6 
7 
8 
9 


10 
11 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 

<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 

<meta http- equiv= "Content- TYpe"”content= "text/html; charset=gb2312" /> 
<title> 注 释 < /title> 

</head> 

<body> 

<?php 


# 创 建 日 期 :2014- 11- 11 
# 创 建 者 : 张 三 
# 本 脚本 用 于 测试 PHP 注释 


echo "这 里 的 echo 0) 语句 
分 为 两 行 输出 。'; 
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20 
21 
22 
23 
24 
25 
26 


/* 
echo "<br /> 这 一 行 单独 显示 .\n\n"; 
*/ 


echo ' 最 后 一 行 .7 //PHP 代码 结束 


< /body> 
</html> 


每 个 脚本 都 应 该 包含 的 最 初 几 条 注释 是 一 个 介绍 性 的 块 ,其 中 列 出 了 创建 日 期 、 修 改 
日 期 .创建 者 、 创 建 者 的 联系 信息 、 脚 本 的 目的 等 。 有 些 人 认为 Shell 风格 的 注释 ( 间 ) 在 
脚本 中 更 醒目 ,因此 这 种 注释 是 最 佳 的 。 

(3) 使 用 多 行 注释 来 注释 第 二 条 echo() 语 句 。 


/* 


echo "<br /> 这 一 行 单独 显示 .\n\n"; 


*/ 


用 / * 和 * /来 包围 的 任何 PHP 代码 块 ,将 不 会 被 PHP 解释 器 运行 ,因此 不 必 将 其 
从 脚本 中 删除 ,之 后 可 通过 删除 注释 标签 ,重新 激活 那 部 分 PHP 代码 。 
(4) 在 最 后 一 条 echo() 语 句 后 面 添加 最 后 一 条 注释 。 


echo ' 最 后 一 行 ."; //PHP 代码 结束 
这 条 最 后 的 注释 说 明了 如 何 把 一 条 注释 放 在 一 行 的 末尾 ,这 是 一 种 惯例 。 


(5) 将 该 文件 另存 为 comments. php, 上 传 到 Web 服务 器 ,然后 在 Web 浏览 器 中 测 
试 它 , 如 图 2-11 所 示 。 


茶 htpyocalhosybeP - CX 上 名 二 白 x 


这 里 的 echo 0 语句 分 为 两 行 输出 。 最 后 一 行 . 


2-11 脚本 2-5 中 的 PHP 注释 不 会 出 现在 Web 页 面 中 


特别 注意 : 不 能 嵌 套 使 用 多 行 注释 (/ * …* /) ,把 一 条 注释 放 在 另 一 条 注释 内 ,因为 
这 会 引起 问题 。 


2. 
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2.2 词法 结构 


2.1 什么 是 变量 


简 言 之 ,变量 是 用 于 临时 存储 值 的 容器 。 这 些 值 可 以 是 数字 、 文 本 或 者 复杂 的 排列 组 
合 。 变 量 在 任何 编程 语言 中 都 居于 核心 地 位 ,理解 它们 是 使 用 PHP 的 关键 所 在 。 

依据 PHP 手册 ,该 语言 中 有 8 种 变量 ,其 中 包括 4 种 标量 ( 单 值 ) 类 型 一 一 布尔 型 
(TRUE 或 FALSE)、 整 型 、 浮 点 型 (小 数 ) 和 字符 串 型 (文本 ); 两 种 非 标量 (多 值 ) 类 
型 一 一 数组 和 对 象 ; 以 及 资源 (用 于 与 数据 库 交互 ) 和 NULL( 它 是 一 种 不 具有 任何 值 的 
特殊 类 型 ) 。 


1 


变量 的 语法 规则 


不 管 创建 什么 类 型 ,PHP 中 的 所 有 变量 都 遵循 依照 以 下 的 语法 规则 : 


变量 的 名 称 ,必须 以 美元 符号 ($ ) 开 头 , 例 如 , $name。 

变量 名 称 可 以 包含 字符 串 .数字 和 下 划 线 的 组 合 ,例如 , $my_reportl 。 

美元 符号 之 后 的 第 一 个 字符 必须 是 字母 或 下 划 线 (不 能 是 数字 ) 。 

PHP 中 的 变量 名 称 是 区 分 大 小 写 的 。 这 意味 着 $name 和 $ Name 是 截然 不 同 的 
变量 。 

可 以 使 用 等 于 号 (= ) ,也 称 为 赋值 运算 符 给 变量 赋值 。 


为 了 开始 使 用 变量 ,我 们 先 利用 几 个 预定 义 的 变量 ,在 运行 PHP 脚本 时 会 自动 设置 
它们 的 值 。 


2. 


打印 出 PHP 预定 义 的 变量 


(1) 在 文本 编辑 器 中 创建 新 的 HTML 文档 (参见 脚本 2-6) 。 


脚本 2-6 这 个 脚本 会 打印 3 个 PHP 预定 义 的 变量 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.wW3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- TYpe"” content= "text/html; charset=gb2312" /> 
<title> 预 定义 变量 < /title> 
</head> 
<body> 
<?Php # 脚 本 2- 6 -predefined.php 


// 创 建 脚 本 中 要 使 用 的 变量 的 简写 版 本 
$file=$_SERVER['PHP SELF']; 
$user=$_SERVER['HTTP USER AGENT']; 
$address=$ _SERVER['REMOTE ADDR']; 


27 


28 


PHP+MySQL 项 目 实例 开发 


16 // 打 印 出 将 被 运行 的 脚本 的 名 称 
17 echo "<p> 您 现在 运行 的 文件 是 :<b> $ file< /b> .< /p> \n"; 


19 // 打 印 出 访问 脚本 的 用 户 的 信息 
20 echo "<p> 您 现在 使 用 的 Web 浏览 器 和 操作 系统 信息 为 :<br /><b>$user< /b><br /> 您 的 IP 
地 址 是 :<br/><b> $address < /b>< /p>\n"; 


2 RS 
23 </body> 
24 </html> 


(2) 创建 脚本 中 要 使 用 的 变量 的 简写 版 本 。 


$file=$_SERVER['PHP SELF']; 

$user=$_SERVER['HTTP_ USER AGENT']; 

$address=$_SERVER['REMOTE ADDR']; 

这 个 脚本 将 使 用 3 个 变量 ,它们 都 来 自 更 大 的 、 预 定义 的 $_SERVER 变量 。$_ 
SERVER 指 大 量 与 服务 器 相关 的 信息 ,如 将 要 运行 的 脚本 的 名 称 ($_SERVERLPHP_ 
SELF ) ,访问 脚本 的 用 户 的 Web 浏览 器 和 操作 系统 ($ _SERVER['HTTP_USER_ 
AGENTJ) 以 及 访问 脚本 的 用 户 的 IP 地 址 ($_SERVER['REMOTE_ ADDR'])。 

用 更 短 的 名 称 创 建新 变量 ,然后 从 $_SERVER 给 它们 赋值 ,这 将 使 得 在 打印 这 些 变 
量 时 更 容易 引用 它们 。 

(3) 打印 出 将 被 运行 的 脚本 的 名 称 。 

echo "<p> 您 现在 运行 的 文件 是 <b> $ file< /b> .< /p> \n"; 


使 用 的 第 一 个 变量 是 $file, 它 具有 与 $_SERVER['PHP_SELF"] 相 同 的 值 。 同 样 ， 
这 个 特殊 变量 总 是 引用 当前 正 被 执行 的 脚本 。 注 意 : 这 个 变量 必须 打印 在 双 引 号 内 ,这 
里 还 利用 了 PHP 换行 符 (\n) , 它 将 在 生成 的 HTML 源 文 件 中 添加 一 个 分 行 符 ; 还 会 添 
加 一 些 基 本 的 HTML 标签 (段落 和 加 粗 ) ,从 而 给 生成 的 页 面 增添 一 些 优雅 的 风格 。 

(4) 打印 出 访问 脚本 的 用 户 的 信息 。 


echo "<p> 您 现在 使 用 的 web 浏览 器 和 操作 系统 信息 为 :<br /><b>$user< /b><br /> 您 的 IP 地 

址 是 :<br/><b>$address </b></p> \n"; 

在 这 里 使 用 了 另外 两 个 变量 。 为 了 重复 在 第 (3) 步 中 所 做 的 工作 ,使 $user 与 $_ 
SERVERLHTTP_USER_AGENTJ] 相 关联 ,并 引用 正 访问 Web 页 面 的 操作 系统 、 浏 览 
器 类 型 和 浏览 器 版 本 。 第 二 个 变量 $address 与 $_SERVER['REMOTE_ADDR'] 相 配 ， 
并 引用 访问 页 面 的 用 户 的 IP 地 址 。 这 里 再 次 使 用 了 换行 符 , 并 且 还 插入 了 一 些 HTML 
分 隔 符 ,使 得 产生 的 Web 页 面具 有 额外 的 行 。 

(5) 将 文件 另存 为 predefined. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 ， 
如 图 2-12 所 示 , 还 可 以 使 用 不 同 的 Web 浏览 器 运行 这 个 脚本 ,测试 真正 的 动态 效果 ， 
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您 现在 运行 的 文件 是 ，/book/ch02/predefined. php. 


您 现在 使 用 的 Web 浏 览 器 和 操作 系统 信息 为 ， 
Jiozilla/4. 0 (compatible; MSIE 7.0; Windows NT 
6.1; WOW64; Trident/5.0; SLCC2; . 

2. 0. 50727; .NET CLR 3. 5- 30729; - 

3. 0. 30729; Media Center PC 6.0; - 

您 的 IP 地 址 是 ， 


127.0.0.1 


-二 


2-12 ”predefined. php 脚本 向 浏览 器 报告 回 关于 脚本 以 及 正 用 于 查看 它 的 Web 浏览 器 的 信息 


[Bm 
$$ @loaho:O 人 和 目 全 7c|| 贺 -B57| 尽 |- 时 |- 人 @ 


园 访问 最 多 | 昭 用 网 址 门 Resizer 
Demerarr services | 


您 现在 运行 的 文件 是 ，/book/ch02/predefined. php. 


您 现在 使 用 的 Web 浏 览 器 和 操作 系统 信息 为 ， 
Mozilla/5.0 (Windows NT 6.1; WOW64; rv:27.0) 
Gecko/20100101 Firefox/27.0 

您 的 IP 地 址 是 ， 

127. 0.0.1 


2-13 ”predefined. php 脚本 使 用 不 同 版 本 的 浏览 器 (对 比 图 2-12) 


创建 变量 时 最 重要 的 考虑 事项 是 使 用 一 致 的 命名 模式 。 在 本 书 中 ,使 用 的 变量 名 称 
用 的 全 都 是 小 写字 母 , 并 且 用 下 划 线 分 隔 单词 ($first_name)。 有 些 程序 员 更 喜欢 使 用 
词 首 大 写字 母 ,例如 , $ FirstName。 

PHP 处 理 变量 的 方式 非常 随意 ,这 意味 着 不 必 初 始 化 变量 或 声明 它们 具体 的 类 型 ， 
并 且 可 以 在 多 种 类 型 之 间 转 换 一 个 变量 ,而 不 会 引发 任何 问题 。 


2.2.2 关于 字符 串 


字符 串 在 Web 应 用 中 非常 常见 ,PHP 提供 了 创建 和 处 理 字 符 串 的 核心 级 支持 。 字 
符 串 (string) 只 是 一 块 用 引号 括 起 来 的 字母 ,数字 、 空 格 \ 标 点 符号 等 ,字符 串 是 任意 长 度 
的 字符 序列 。 下 面 列 出 的 全 都 是 字符 串 : 

' 张 三 ' 

"Hello Word!™ 
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"4000" 
"2014 年 12 月 25 日 " 


为 了 建立 一 个 字符 串 变 量 , 可 以 给 一 个 有 效 的 变量 名 赋予 一 个 字符 串 值 : 


$first name=' 张 '; 
$today='2014 年 12 月 25 日 '; 


为 了 打印 字符 串 的 值 , 可 以 使 用 echoO 〇 或 print() : 


echo $ first name; 


为 了 在 某 种 环境 内 打印 出 字符 串 的 值 , 可 以 使 用 双 引 号 : 


echo "你 好 ,S first_name"7 


下 


使 用 字符 串 


(1) 在 文本 编辑 器 中 创建 一 个 新 的 HTML 文档 ,以 及 PHP 标签 (参见 脚本 2-7) 。 


脚本 2-7 在 这 个 介绍 性 的 脚本 中 创建 字符 串 变 量 并 把 它们 的 值 发 送 给 Web 浏览 器 。 


19 
20 
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< !DOCTYPE html PUBLIC "~ //W3C/V/DTD XHTML 1.0 Transitional//EN” 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312"”/> 
<title> 字 符 串 < /title> 
</head> 
<body> 
<?php # 脚 本 2- 7- strings.php 


// 创 建 变量 

$first_name= ' 斯 提 芬 妮 '; 
$1ast_name= ' 梅 尔 '; 
$book= ' 暮 光 之 城 '7 


// 打 印 变量 
echo "小 说 <i>$book< /i> 由 $first_name $last_name 所 著 . "7 


?> 
</body> 
</htm> 


(2) 在 PHP 标签 内 ,创建 3 个 变量 。 
$first_name= ' 斯 提 芬 妮 '; 


$last_name= ' 梅 尔 '; 


$book= ' 莫 光 之 城 '; 
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在 这 个 基本 的 示例 中 ,创建 了 $ first_name、$ last_name 和 $book 这 3 个 变量 ,随后 
将 会 在 一 条 消息 中 打印 出 它们 。 
(3) 创建 echo 〇 语句 。 


echo "小 说 <i>$book< /i> 由 $first name $last name 所 著 。"; 


这 个 脚本 所 做 的 全 部 工作 是 基于 已 建立 的 3 个 变量 打印 出 一 份 作者 身份 声明 。 其 中 
插入 了 很 少 的 HTML 格式 化 效果 (斜体 ) ,使 之 实现 强调 效果 。 记 住 在 这 里 为 要 打印 的 
变量 值 相应 地 使 用 双 引 号 (在 本 章 末 尾 将 更 详细 地 讨论 双 引 号 的 重要 性 ) 。 

(4) 将 该 文件 另存 为 strings. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 ， 
如 图 2-14 所 示 。 


全 多 hupy/ocalhosybe PD - OX 


小 说 宫 光 之 婚 由 斯 提 苏 妮 梅 尔 所 著 。 


图 2-14 得 到 的 Web 页 面 基 于 打印 出 3 个 变量 的 值 


(5) 作为 测试 ,可 更 改 3 个 变量 的 值 .然后 再 次 运行 脚本 ,如 图 2-15 所 示 。 


we/eotonm Dox] sn «| 


小 说 咒 三 届 由 易中天 所 著 。 


图 2-15 通过 改变 脚本 中 的 变量 来 改变 脚本 的 输出 


如 果 把 另 一 个 值 赋予 现 有 的 变量 (比如 , $ book) ,新 的 值 就 会 重 写 旧 的 值 。 例 如 ， 


$book= ' 募 光 之 城 ' 
$book= ' 品 三 国 '; // 现 在 $book 的 值 变 为 ' 品 三 国 ' 


2. 连接 字符 串 

连接 字符 串 是 创建 动态 Web 站 点 时 的 一 个 重要 工具 , 像 是 为 字符 串 增加 的 一 种 功 
能 ,可 以 使 用 连接 运算 符 即 句点 (. ) 来 执行 它 。 

$city= "南京 '; 

$state=' 江 苏 '; 

$address=$ state . Scity; 
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$ address 变量 的 值 现 在 是 “江苏 南京 "。 为 了 对 其 进行 改进 ,还 可 以 将 逗号 和 空格 添 
加 到 字符 串 混合 中 : 


$address=$ state . ',' . $city; // 生 成 "江苏 ,南京 " 
连接 还 可 以 处 理 字符 串 或 数字 。 下 面 两 条 语句 将 会 产生 相同 的 结果 : 


$address=$ state . ',' . $city . ' 210000"7 
$address=$ state . ',' . $city .' ' . 210000; 


连接 常 与 字符 串 变 量 一 起 使 用 ,在 后 面 的 章节 中 构建 数据 库 查询 时 会 大 量 用 到 它 。 

现在 修改 strings. php 脚本 ,以 使 用 连接 这 个 新 工具 。 

(1) 在 文本 编辑 器 中 打开 strings. php( 参 见 脚 本 2-7)。 

(2) 在 创建 了 $first_name 和 $last_name 变量 之 后 (第 12 行 和 第 13 行 ) ,添加 如 下 
语句 (参见 脚本 2-8) 。 


$author=$ first name . ' ' . $1last name; 


脚本 2-8 连接 允许 轻松 操纵 字符 串 , 如 通过 名 字 和 姓氏 的 组 合 创 建 作者 的 姓名 。 


< !DOCTYPE html PUBLIC "~ //W3C//DID XHTML 1.0 Transitional//EN" 
"http://www.w3.0org/TR/xhtml1/DTD/xhtmll- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 


1 
2 
3 
4 
5 <meta http-equiv= "Content- Type" content="text/html; charset=gb2312"”/> 
6 ”<title> 连 接 字符 串 < /title> 

7 </head> 

8 <body> 

9 ”<?php # 脚 本 2- 8- concat.php 

11 // 创 建 变量 

12 $first_name=' 史 提 芬 '; 

13 $last name=' 金 '; 

14 $author=$ first name . '' .$1ast name; 


16 $book=' 肖 申 克 的 救赎 '; 


18 // 打 印 变 量 
19 echo "小 说 <i>$book< /i> 由 $author 所 著 ."? 


21 号 > 
22 </body> 
23 </htm> 


如 连接 的 示范 那样 ,将 会 创建 一 个 新 变量 $ author, 它 连接 了 两 个 现 有 的 字符 串 和 它 
们 之 间 的 一 个 空格 。 
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(3) 更 改 echo() 语 句 , 使 用 这 个 新 变量 。 
echo "小 说 <i>$book< /i> 由 $author 所 著 。"; 


因为 把 两 个 变量 变 成 了 一 个 变量 ,所 以 应 该 相应 地 改变 echo() 语 句 。 
(4) 将 文件 另存 为 concat. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 , 如 
图 2-16 所 示 。 


小 说 六 第 站 和 式 用 由 史 提 芬 金 所 著 。 


图 2-16 连接 的 最 终结 果 对 用 户 从 来 不 是 显而易见 的 (对 比 图 2-14 和 图 2-15) 


PHP 具有 许多 特定 于 字符 串 的 有 用 函数 ,功能 十 分 强大 ,我 们 将 在 后 续 章 节 中 一 一 
进行 详细 介绍 。 


2.2.3 关于 常量 


常量 是 PHP 中 一 种 特定 的 数据 类 型 ,与 变量 不 同 的 是 ,常量 在 整个 脚本 中 都 会 保持 
它们 的 初始 值 。 事 实 上 ,一 旦 设置 了 常量 ,就 不 能 更 改 它 的 值 。 常 量 可 以 被 赋予 任意 单 
值 . 如 一 个 数字 或 字符 串 ,而 不 能 被 赋予 类 似 数组 这 样 的 多 值 。 

1. 创建 常量 

要 创建 常量 ,可 以 使 用 define() 函数 ,而 不 是 用 于 变量 的 赋值 运算 符 ( 一 ) 。 

define('NAME', ‘value'); 

注意 : 一 条 经 验 法 则 是 ,全 都 使 用 大 写字 母 来 命名 常量 ,尽管 并 非 必须 如 此 。 最 重要 
的 是 ,常量 不 会 像 变量 那样 使 用 美元 符号 ,因为 从 技术 上 讲 常量 不 是 变量 。 

2. 打印 常量 

打印 常量 也 需要 特殊 的 语法 : 

define ('USERNAME',' 张 三 '); 

echo 'Hello, ' . USERNAME; 

不 能 使 用 echo " Hello，USERNAME" 打 印 常量 ,因为 PHP 只 会 打印 出 Hello， 
USERNAME ,而 不 会 打印 出 USERNAME 常量 的 值 ,因为 没有 变量 标识 符 $ (美元 符 
号 ) 告 诉 PHP: USERNAME 是 不 同 于 字面 量 文本 的 任何 内 容 。 

PHP 运行 时 利用 了 几 个 预定 义 的 常量 ,这 与 本 章 前 面 使 用 的 预定 义 变量 非常 相像 。 
这 些 常量 包括 PHP_VERSIONCPHP 运行 的 版 本 ) 和 PHP_OS( 服 务 器 的 操作 系统 ) 。 
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3. 


使 用 常量 


(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 2-9) 。 


脚本 2-9 常量 是 在 PHP 中 可 以 使 用 的 另 一 种 数据 类 型 , 它 不 同 于 变量 。 


< !DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtmll1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- TYPe”content= "text/html; charset=gb2312" /> 
<title> 常 量 < /title> 
< /head> 
<body> 
<?Pphp # 脚 本 2- 9- constants.php 


// 将 今天 的 日 期 设置 为 常量 
define('TODRY'，'2014 年 12 月 25 日 "); 


// 打 印信 息 ,使 用 预定 义 常量 和 自 定义 的 ToDaY 常量 
echo ' 今 天 日 期 为 : ' . TODAY . '。<br /> 服务 器 使 用 的 PHP 版 本 为 :<b> ' . PHP_VERSION , 
</b> ,使 用 的 操作 系统 为 : <b>' . PHP_ 0S . '</b> 。' 


EE 
< /body> 
</html> 


(2) 创建 一 个 新 的 日 期 常量 。 


define ('TODRY'，'2014 年 12 月 25 日 '); 


人 们 普遍 认为 使 用 常量 的 意义 不 大 ,但 是 ,这 个 示例 将 说 明 其 重要 性 。 在 后 续 章节 
中 ,将 使 用 常量 来 存储 数据 库 访 问 信息 。 

(3) 打印 出 日 期 .PHP 版 本 以 及 操作 系统 信息 。 

echo ' 今 天 日 期 为 : ' . TODAY . '。<br /> 服务 器 使 用 的 PHP 版 本 为 :<b> ' . PHP_VERSION . 

"< /b> ,使 用 的 操作 系统 为 : <b>' . PHP os . '</b>。'; 

因为 常量 不 能 打印 在 双 引 号 内 ,缺少 美元 符号 将 使 它们 被 视 作 大 写 的 文本 ,所 以 使 用 
连接 运算 符 来 创建 echo() 语 句 。 

(4) 将 文件 另存 为 constants. php, 上 传 到 Web 服务 器 .并 在 Web 浏览 器 中 测试 它 ， 
如 图 2-17 所 示 。 


2.2.4 关于 数字 


在 讨论 变量 时 ,PHP 具有 整 型 和 浮 点 型 (小 数 ) 数 字 类 型 。 但 是 ,经 验证 明 , 这 两 种 类 


全 IErewmeernwwp-cxje 


今天 日 期 为 ， 2014 年 12 月 25 日 。 
服务 部 使 用 的 PIP 版 本 为 ， 5. 2. 6 ， 使 用 的 操作 系统 为 ， 
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图 2-17 使 用 PHP 的 常量 


型 可 以 归 类 到 一 般 的 数字 (number) 之 下 ,在 大 多 数 情况 下 ,不 会 有 任何 差别 。PHP 中 有 


效 的 数字 类 型 的 变量 可 以 是 如 下 形式 


8 

3.14 
10980843985 
-4.2398508 
4.4e2 


注意 : 这 些 值 永远 不 会 用 引号 括 起 来 ,如 果 这 样 做 ,它们 就 是 具有 数值 的 字符 串 。 此 
外 ,数字 被 假定 为 正 , 除 非 在 其 前 面 放置 一 个 负 号 (一 ) 。 
对 数字 除了 可 以 使 用 标准 的 算术 运算 符 ( 参 见 表 2-1) 之 外 ,还 可 以 使 用 许多 函数 。 
在 这 里 先 介 绍 的 round() 和 number_format() 两 个 函数 。 前 者 用 于 把 小 数 四 舍 五 入 为 最 


接近 的 整数 : 
表 2-1 标准 数学 运算 符 
运 算 符 含 本 运 算 符 含 党 
EE 加 法 % 取 模 
2 减法 叭 二 增 量 
be 乘法 ee 减 量 
# 除法 
$n=3.14; 
$n=round ($n); /1/3 
或 者 把 小 数 四 会 五 人 到 指定 的 位 数 : 
$n=3.142857; 
$n=round ($n, 3); //3.143 


number_format() 函 数 用 于 把 一 个 数字 转换 成 更 普遍 地 书写 的 版 本 ,并 使 用 逗号 把 


它 分 成 “三 位 一 组 ”。 例 如 : 


$n=20943; 
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$n=number format ($n); 


这 个 函数 还 可 以 设置 小 数 点 的 指定 位 数 : 


//20,943 


$n=20943; 


$n=number format ($n, 2); //20,943.00 


2.2.5 单 引 号 与 双 引 号 


在 PHP 中 ,理解 单 引 号 与 双 引号 有 什么 区 别 是 重要 的 。 如 本 章 的 示例 中 所 示 , 可 以 
使 用 echo() 或 print() 语 句 , 但 是 ,这 两 个 语句 以 及 使 用 它们 的 原因 之 间 有 一 个 关键 的 区 
别 。 之 前 已 详细 说 明了 何 时 应 该 使 用 它们 ,但 是 ,现在 将 更 明确 地 定义 它们 的 使 用 模式 。 

1. 单 引号 和 双 引 号 区 别 

在 PHP 中 ,封闭 在 单 引号 内 的 值 将 照 字 面 意义 进行 处 理 ,而 封闭 在 双 引 号 内 的 值 则 
将 被 解释 。 换 句 话 说 ,把 变量 和 特殊 字符 (参见 表 2-2) 放 在 双 引 号 内 将 导致 打印 出 它们 
表示 的 值 ,而 不 是 它们 的 字面 值 。 例 如 : 

$var= 'test'; 

echo "var 的 值 是 $ var"; 

echo "var 的 值 是 $var'; 

使 用 一 个 转 义 的 美元 符号 : 

echo "\$var 的 值 是 $var"; // 将 打印 出 :$var 的 值 是 test 

echo '\$var 的 值 是 $var'; // 将 打印 出 :\$ var 的 值 是 $var 

正如 这 些 示例 所 说 明 的 , 双 引号 将 用 变量 的 值 (test) 代 替 它 的 名 称 ($ var) ,并 用 特殊 
字符 表示 的 值 ($ ) 代 替 它 的 代码 (\$)。 单 引号 总 是 准确 地 打印 输入 的 内 容 , 除 了 转 义 的 
单 引 号 (\7 和 转 义 的 反 斜 枉 (\\) 之 外 ,它们 将 分 别 被 打印 为 一 个 单 引 号 和 一 个 反 斜 杠 。 

表 2-2 当 在 双 引号 内 使 用 这 些 字符 时 ,它们 具有 特殊 的 含义 


// 将 打印 出 :var 的 值 是 test 
// 将 打印 出 :var 的 值 是 $ var 


转 义 字符 的 代码 转 义 字符 的 含义 转 义 字符 的 代码 转 义 字符 的 含义 
双 引 号 \r 回 车 符 
Vv 单 引号 \t 制 表 符 
\ 反 斜 杠 \$ 美元 符号 
\n 换行 符 


使 用 反 斜 杠 对 字符 进行 转 义 是 一 个 重要 的 概念 。 当 给 变量 赋值 或 者 将 数据 作为 参数 
发 送 给 函数 时 ,上 述 规 则 适用 于 echo() 或 print() 语 句 内 使 用 的 任何 引号 。 

2. 使 用 单 引 号 和 双 引 号 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 2-10) 。 


脚本 2-10 本 脚本 演示 了 何 时 以 及 如 何 使 用 一 种 引号 而 不 使 用 另 一 种 引号 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN™" 
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2 "http://www.w3.0rg/TR/xhtml1/DID/xhtml1- transitional.dtd"> 
3 <html xmlns= "http://www-w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv= "Content- Type" content="text/html; charset=gb2312" /> 
6 ”<title> 单 引号 与 双 引 号 < /title> 

7 </head> 

8 <body> 

9 ”<?php # 脚 本 2- 10- quotes.php 

10 ”// 定 义 变量 

11 $filel='c:\windows\remove.ini'; 

12 $file2="c:\\windows\\remove.ini"; 

13 $link="I'mhappy."; 


15 “// 打 印 HTML 代码 

16 $htmll='<ahref="' . $filel .'">' .$link. '</a>'; 
17 $html2="<a href=\"$file2\">$1ink</a>"; 

18 echo $htmll . "<br />\n"; 

19 echo $html2 . '<br />'; 


21 ?> 

22 </body> 

23 </html> 

(2) 创建 3 个 变量 。 


$filel='c:\windows\remove.ini'; 

$ file2="c:\\windows\\remove.ini"; 

$1ink="I'm happy." ; 

可 以 看 到 , 反 斜 杠 在 字符 串 中 有 它 的 特殊 含义 , 当 需 要 在 字符 串 中 包含 反 斜 杠 本 身 
时 ,需要 在 该 符号 前 面 多 加 一 个 反 斜 杠 或 者 如 $filel 变量 定义 方式 ,直接 将 字符 串 封闭 
在 单 引 号 内 。 

(3) 打印 出 两 个 HTML 的 超 链接 。 


$htmll='<a href="' . $filel . '">' .$link. '</a>'; 
$html2="<a href=\"$ file2\">$ 1ink< /a> "7 

echo $htmll . "<br /> \n"7 

echo $html2 . "<br />'; 


这 里 演示 了 两 种 方式 的 引号 效果 ,$htmll 变量 使 用 单 引号 和 连接 运算 符 , $ html2 
使 用 反 斜 杠 转 义 的 方式 。 尤 其 需要 注意 的 是 换行 符 (\n) 必 须 括 在 双 引 号 中 才能 生效 。 

在 标准 HTML 语言 中 双 引 号 常 被 用 来 表示 标签 内 的 属性 值 ( 现 在 很 多 浏览 器 具备 
较 强 的 容错 功能 ,允许 在 HTML 中 用 单 引 号 甚至 不 用 引号 表示 字符 串 ) 。 

(4) 将 文件 另存 为 quotes. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 , 如 
图 2-18 所 示 ,并 查看 源 代码 检测 PHP 运行 结果 .如 图 2-19 所 示 。 
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(DO roo. p - ox 


2-18 ”quotes. php 在 浏览 器 中 运行 结果 


IO 30 ii 1 Lr 60 
工 <!DOCTYPE html PUBLIC "~-//W3C//DTD XHTNL 1.0 Transitional//EN" 
2 "http://weu.w3.org/TR/xhtml1/DTD/xhtml1-transitional. dtd"> 
3 <html xmlns="http://vew.w3.org/1999/xhtml"> 
4 <head> 
5 <meta http-equiv="Content-Type” content="text/html; charset=gh2312" /> 
5 <title> 单 引导 与 双 引 号 </title> 
7 </head> 
8 <body> 
9 <a href="c:\windows\remove. ini">I'm happy.</a><br /> 
0 <a href="c:\windows\remove. ini">I'm happy.</a><br /></body> 


图 2-19 quotes. php 的 HTML 源 代码 


由 于 PHP 将 试图 找 出 那些 需要 将 其 值 插入 到 双 引 号 内 的 变量 ,所 以 从 理论 上 讲 , 使 
用 单 引号 要 快 一 些 。 但 是 ,如 果 需 要 打印 一 个 变量 的 值 , 则 必须 使 用 双 引 号 。 

因为 有 效 的 HTML 常常 包括 许多 用 双 引 号 括 住 的 属性 ,所 以 当 利 用 PHP 打印 
HTML 时 ,使 用 单 引号 最 容易 。 

echo '<table width= "80%" border= "0" cellspacing="2" 

cellpadding= "3" align= "center"> "7 

如 果 想 使 用 双 引 号 打印 出 这 段 HTML 代码 ,将 不 得 不 对 字符 串 中 的 所 有 双 引 号 进 
行 转 义 。 

echo "<table width=\"80$\" border=\"0\" cellspacing=\"2\" 

cellpadding=\"3\" align=\"center\">"; 


2.3 项 目 训 练 内 容 管 理 系 统 CMS 首页 设计 


2.3.1 项 目 说 明 


制作 一 个 最 简单 的 内 容 管理 系统 (Content Management System,CMS) 首 页 ,可 以 是 
新 闻 发 布 系统 或 博客 (Blog) 的 首页 ,该 页 面 中 包含 一 些 特定 信息 ,例如 ,网 站 名 称 、 文 章 名 
称 、 站 长 信息 等 ,这 些 特定 信息 有 两 个 特点 : 首先 ,实际 应 用 中 ,如 果 用 户 登 录 的 身份 或 时 
间 地 点 不 同 , 这 些 信息 会 有 所 变化 ;其 次 , 当 某 一 用 户 访问 该 站 点 的 多 个 页 面 时 ,这 些 信息 
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在 每 一 页 中 显示 的 内 容 和 效果 又 是 相同 的 。 
2.3.2 设计 思路 


分 析 需 求 可 知 ,动态 网 站 中 的 内 容 一 定 是 存储 在 各 种 变量 中 的 ,这 些 变量 有 的 来 源 于 
数据 库 , 有 的 来 源 于 表单 提交 等 。 在 实际 开发 过 程 中 ,只 需 设 计 好 静态 页 面 的 布局 效果 ， 
然后 将 各 个 变量 分 别 打印 在 特定 的 位 置 , 就 可 实现 界面 美化 的 动态 页 面 。 

为 此 ,可 以 借鉴 任何 一 个 网 页 模板 或 已 成 型 的 CMS 项 目 , 先 完成 静态 页 面 布局 ,再 
添加 动态 PHP 代码 。 


2.3.3 设计 过 程 


首先 ,设计 一 个 最 简单 静态 页 面 ,包含 一 定 的 页 面 布局 与 CSS, 并 设计 了 多 个 变量 , 打 
印 在 页 面 的 各 个 位 置 ,灵活 实现 Web 页 面 的 动态 效果 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 2-11) 。 


脚本 2-11 本 脚本 演示 了 某 站 点 页 面 的 显示 模板 ,并 使 用 PHP 变量 来 灵活 实现 动态 
效果 。 


<? 
// 全 站 通用 变量 
S$ MyEmail= "adming domain.com"; 
$MyEmailLink= "<a href= \"mailto:$ MyEmail\"> $ MyEmail< /a> "; 
$ MyName= "Alex Wang"; 
$MySiteName= $ MyName ."'s Blog"; 
$PageTitle= "连接 运算 符 的 使 用 "7 
?> 
< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.orgVTR/ 
xhtmlL1/DTD/xhtmll-transitional.dtd"> 
10 <html xmlns= "http://www.w3.org/1999/xhtml"> 
11 <head> 


ownaowawm wh 


12 <meta http- equiv= "Content- TYpe"”content= "text/htm1; charset=gb2312" /> 
13 <title><?php echo "$MySiteName -$PageTitle"” ?>< /title> 

14 <style type= "text/css"> 

15 hl,pffont- family:Tahomay} 

16 </style> 

17 </head> 


19 <body> 

20 <hl><?php echo "$MySiteName"; ?></hl> 
21 <hr /> 

22 ”正文 信息 ! 

23 <hr /> 
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24 <palign="center"> 
25 Copyright scopy; by <?php echo "$MyName ($MyEmailLink)"; ?>, 2014 
26 </p> 

27 </body> 

28 </html> 


(2) 在 HTML 代码 开始 之 前 , 先 定义 了 5 个 变量 ,分 别 定义 了 站 长 邮箱 、 邮 箱 超 链 
接 、 站 长 昵称 、 站 点 名 称 、 页 面 标题 等 信息 。 

$ MyEmail= "admin@ domain.com"; 

$MyEmailLink="<a href=\"mailto:$ MyEmail\"> $MyEmail< /a> "7 

$ MyName= "Alex Wang"; 

$ MySiteName= $ MyName ."'s Blog"; 

$ PageTitle= "连接 运算 符 的 使 用 "; 

(3) 从 二 title 二 开始 , 即 开始 动态 打印 已 定义 好 的 各 个 变量 。 

(4) 将 文件 另存 为 website. php ,上传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 , 如 
图 2-20 所 示 。 


| 


IS @ htpylocalhosybeP - © X||@ Alex Wang's Blog - . x 
Alex Wang's Blog 


正文 信息 ! 


Copyright © by Alex Wang (admin@domain.com) , 2014 


2-20 site. php 在 浏览 器 中 的 运行 结果 


本 章 小 结 


本 章 介 绍 了 PHP 语言 的 基础 语法 ,包括 PHP 的 嵌入 方法 .变量 和 常量 ,数据 类 型 、 
数字 和 字符 转 义 。 学 习 这 些 基础 知识 将 会 为 以 后 更 好 地 掌握 PHP 打下 基础 。 


重点 回顾 


1. 在 Web 页面 中 使 用 PHP 标签 区 入 PHP 代码 。 
2. 如 何 输 出 数据 到 Web 页 面 及 空白 处 理 。 

3. PHP 的 变量 与 常量 定义 。 

4. 使 用 字符 串 与 特殊 字符 处 理 。 
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本 章 实 训 


【 实 训 1 
新 建 hello. php ,练习 如 何在 HTML 中 骨 入 PHP 脚本 ,并 使 用 echo 命令 输出 任意 字 
符 串 ,并 添加 网 页 标题 和 相关 注释 。 网 页 运行 效果 如 图 2-21 所 示 。 


图 2-21 hello. php 的 运行 效果 


【 实 训 2】 

新 建 string. php, 练 习 字 符 串 变 量 单 、 双 引号 使 用 及 转 义 符 使 用 ,分 别 设 定 如 下 5 个 
变量 ,变量 值 替 换 为 你 的 真实 信息 ,并 在 网 页 中 通过 echo 输出 ,效果 如 图 2-22 与 图 2-23 
所 示 。 


(| httpi//ocalhostbe DP ~ © x 


我 的 测试 服务 器 地 址 ，http: 区 拒 3 广 人/ 
网 页 保存 路 径 ，C:\VAppServ\www\ 学 号 文件 夫 


| 3 0111111 
我 的 英文 名 ，Ja 


今日 心情 : mhappy! 


图 2-22 ”string. php 的 运行 效果 


<body> 
我 的 测试 服务 器 地 址 ，http://1ocalhost/ 学 号 文件 夹 /<br /> 
网 页 保存 路 答 ，c:\Appserv\Vwww\ 学 号 文件 夹 <br /> 
我 的 学 号 ，011111i<br /> 

我 的 英文 名 Jack<br /> 
今日 心情 :I'm happy! 

</body> 


图 2-23 生成 的 网 页 HTML 代码 片段 ,注意 HTML 源 代码 的 换行 


【 实 训 3】 

新 建 table. php ,练习 预定 义 常 量 和 预定 义 变量 的 使 用 .请 参见 脚本 2-6 及 脚本 2-9， 
获取 服务 器 及 用 户 客户 端的 信息 ,网 页 运行 效果 参考 如 图 2-24 所 示 。 

具体 要 求 : 

(1) 今天 的 日 期 使 用 自 定义 常量 ,请 参见 脚本 1-10。 
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(2) 信息 以 表格 形式 输出 ,如 图 2-24 所 示 。 
(3) 有 关 脚 本 的 相对 及 绝对 路 径 的 预定 义 变量 请 参考 PHP 中 文 帮助 。 


欢迎 光临 本 站 ， 今 天 是 ，2014 年 12 月 25 日 
您 的 信息 如 下 表 ， 


操作 系统 wNNT 
14.0 (compatible; MSIE 7.0; 

NT 6.1; WOW64; Trident/5.0; 
皮 作 系统 与 浏览 器 版 本 ; .NET CLR 2.0.50727; .NET CLR 
.5.30729; .NET CLR 3.0.30729; Media 

PC 6.0; .NET4.0C) 


IP 地 址 127.0.0.1 
PHP 版 本 5.2.6 
脚本 的 相对 路 径 名 /book/ch02_ex/table.php 
脚本 的 绝对 路 径 名 C:/AppServ/www/book/ch02_ex/table.php| 


图 2-24 table. php 的 运行 结果 
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任何 PHP 脚本 都 是 由 一 系列 语句 构成 的 。 一 条 语句 可 以 是 一 个 赋值 语句 一 个 函 
数 调用 一 个 循环 .一 个 条 件 语 句 或 者 甚至 是 一 个 什么 也 不 做 的 语句 ( 空 语 句 )。 语 句 通 常 
以 分 号 结束 。 此 外 ,还 可 以 用 花 括 号 将 一 组 语句 封装 成 一 个 语句 组 。 语 句 组 本 身 可 以 当 
作 是 一 行 语句 。 


3.1 条件 语句 与 运算 符 


3.1.1 认 条 件 语句 


条 件 语句 像 变 量 一 样 是 程序 设计 的 一 个 组 成 部 分 ,大 多 数 人 都 熟悉 它们 的 某 一 种 或 
另外 一 种 形式 。 动 态 Web 页 面 经 常 需要 使 用 条 件 语 名 ,依据 设置 的 准则 来 改变 脚本 的 
行为 。 

PHP 用 于 创建 条 件 语 句 的 3 个 主要 术语 是 : if、else 和 elseif( 它 也 可 以 写作 两 个 单 
词 : else if) 。 

在 PHP 中 ,条 件 语句 有 三 种 形式 ,每 个 条 件 语 句 都 包含 有 一 个 让 子 句 。 

1. 证 结构 

证 (条 件 ){ 

语句 
} 
首先 判断 “条 件 ”, 如 果 条 件 为 真 (true) , 则 执行 “语句 ”; 如 果 条 件 为 假 (False) ,将 忽略 


2. if-else 结构 


条 件 语句 的 第 二 种 形式 是 if-else, 除 了 让 语句 之 外 ,还 加 上 了 else 语句 , 它 可 以 在 计 
语句 中 的 表达 式 的 值 为 False 时 执行 。 


证 (条 件 ){ 
语句 1 
Jelse{ 
语句 2 
} 
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首先 判断 “条 件 ”, 如 果 条 件 为 真 (true) , 则 执行 “语句 1”; 如 果 条 件 为 假 (False) , 则 执 
行 “ 语 句 2”。 

3. if-elseif 结构 

条 件 语 句 的 第 三 种 形式 是 让 elseif,elseif 是 这 和 else 的 组 合 。 和 else 一样, 它 延 伸 了 
让 语句 ,可 以 在 原来 的 让 表达 式 值 为 False 时 执行 不 同 语句 。 但 是 和 else 不 一 样 的 是 , 它 
仅 在 elseif 的 条 件 表达 式 值 为 True 时 执行 语句 ,语法 如 下 : 

证 (条 件 1) { 

语句 1 

Jelseif (条 件 2) { 

语句 2 
Jelseif (条 件 3) { 
Jelse{ 

语句 nn 

} 

如 果 条 件 为 真 , 则 将 执行 其 下 面 的 花 括 号 ({)) 中 的 代码 。 如 果 条 件 不 为 真 ,PHP 将 
继续 往 下 执行 。 首 先 判断 “条 件 1”, 如 果 * 条 件 1” 为 真 (true) , 则 执行 “语句 1”; 如 果 “ 条 件 
1” 为 假 (False) , 则 判断 “条 件 2”"。 如 果 * 条 件 2 为 真 (true) , 则 执行 “语句 2”; 如果“ 条 件 
2” 为 假 (False) , 则 判断 “条 件 3”, 以 此 类 推 ,如 果 所 有 的 表达 式 的 值 都 为 False, 则 执行 “ 语 
名 n"。 

这 个 过 程 将 继承 一 一 可 以 根据 需要 使 用 多 个 elseif 子 句 直至 PHP 遇 到 else, 此 
时 将 自动 执行 它 ;或 者 如 果 没 有 else, 则 PHP 将 执行 到 条 件 语句 终止 为 止 。 

因此 ,总 是 把 else 放 在 最 后 ,并 将 其 视 作 默认 的 动作 ,除非 满足 特定 的 条 件 ,这 样 做 
是 重要 的 。 

PHP 中 条 件 为 真 的 原因 有 许多 种 。 下 面 是 常见 的 条 件 为 真 的 情况 : 

。 $ var, 如 果 变量 $ var 非 空 ,也 就 是 具有 非 0 值 、. 空 字符 串 或 NULL, 则 条 件 为 真 。 

。 isset( $ var) ,如 果 变 量 $ var 具有 不 同 于 NULL 的 任何 值 , 包 括 0 或 空 字符 串 , 则 

条 件 为 真 。 

。 TRUE,true、True 等 。 

在 上 述 为 真 条 件 中 ,引入 了 一 个 函数 isset()。 这 个 函数 用 于 检查 一 个 变量 是 否 被 设 
置 ,这 意味 着 它 具 有 一 个 不 同 于 NULL 的 值 。 还 可 以 连同 符号 一 起 使 用 比较 和 逻辑 运算 
符 ( 参 见 表 3-1 和 表 3-2) ,来 建立 更 复杂 的 表达 式 。 

在 编写 条 件 语句 时 ,经 常会 使 用 如 表 3-1 所 示 的 比较 和 逻辑 运算 符 。 

比较 运算 符 ,允许 对 两 个 值 进行 比较 .PHP 常用 的 比较 运算 符 如 表 3-1 所 示 。 

如 果 比 较 一 个 整数 和 字符 串 , 则 字符 串 会 被 转换 为 整数 。 如 果 比 较 两 个 数字 字符 串 ， 
则 作为 整数 比较 。 

PHP 常用 的 逻辑 运算 符 如 表 3-2 所 示 。 
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表 3-1 算术 运算 符 
符 ”号 含义 示 例 
== 等 于 $a== $b 
一 一 一 全 等 ,等 于 且 类 型 相同 $a===$b 
4 一 不 等 $al=$b 
!== 不 全 等 ,不 等 或 类 型 不 同 $al== $b 
> 大 于 $a>$b 
< 小 于 $a<$b 
>= 大 于 或 等 于 $a>=$b 
<= 小 于 或 等 于 $a<=$b 
表 3-2 逻辑 运算 符 
符 号 含 义 示 例 
! 逻辑 非 1$a 
&& 逻辑 与 $a&&$b 
11 逻辑 或 $all$b 
XOR 逻辑 异 或 $aXOR $b 


“与 "和 “或 "有 两 种 不 同形 式 运 算 符 , 它 们 运算 的 优先 级 不 同 ,&& 和 | | 优先 级 高 。 


4. 


使 用 条 件 语句 


(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 3-1) 。 


脚本 3-1 代码 中 的 条 件 语句 允许 依据 不 同 的 条 件 执行 不 同 的 行为 。 


oo aonmrnopDpr 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- TYpe"” content= "text/html; charset=gb2312" /> 
<title> 条 件 语句 < /title> 
</head> 
<body> 
<?php # 脚 本 3- 1-if.php 


// 初 始 化 $gender 变量 
$gender= 男 } 


// 根 据 变量 的 值 输出 欢迎 内 容 
if (isset ($ gender)) { 
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15 if($gender==' 男 '){ 


16 echo '<p><b> 您 好 ,先生 !< /b>< /p> '; 

17 } elseif ($ gender==' 女 '){ 

18 echo '<p><b> 您 好 ,女士 !</b>< /p> '; 

19 } else { //$ gender 变量 的 值 无 效 
20 echo '<p><b> 请 您 重新 输入 您 的 性 别 。< /b>< /p> '; 
21 } 

22 Jelse{ 

23 echo '<p><b> 您 还 未 输入 您 的 性 别 。< /b>< /p> '; 

24 } 

25 

26 ?> 

27 </body> 

28 </htm> 


在 HTML 表单 验证 输入 时 ,经 常会 使 用 单 选 按 钮 . 复 选 框 或 选项 菜单 等 控件 ,此 时 
使 用 条 件 语 句 是 一 种 简单 ` 有 效 的 方式 。 如 果 用 户 选 中 了 任何 一 个 性 别 单 选 按钮 ,那么 传 
递 到 后 台 的 变量 将 具有 一 个 值 , 这 时 可 以 使 用 isset() 函数 来 判断 条 件 是 否 为 真 。 如 果 条 
件 不 为 真 ,那么 指示 它 没有 值 或 无 效 。 

(2) 在 第 一 个 条 件 语 名 (isset() ) 中 ,和 嵌 套 了 另 一 个 条 件 语 句 ,基于 $ gender 的 值 打 印 
一 条 消息 。 


if($gender==' 男 '){ 
echo '<p><b> 您 好 ,先生 !< /b>< /p>'; 
} elseif ($ gender==' 女 '){ 
echo '<p><b> 您 好 ,女士 !< /b>< /p>'; 
} else { 
echo '<p><b> 请 您 重新 输入 您 的 性 别 。</b></p> '; 
} 
这 个 if-elseif-else 条 件 语 句 会 查看 $ gender 变量 的 值 ,并 针对 每 一 种 可 能 性 打印 一 
条 不 同 的 消息 。 双 等 于 号 (二 三 ) 指 相等 ,而 单个 等 于 号 (==) 则 用 于 赋值 , 记 住 这 一 点 非常 
重要 。 这 个 区 别 很 重要 ,因为 条 件 $ gender 二 二 ' 男 ' 可 能 为 真 ,也 可 能 不 为 真 ;但 是 
$ gender 一 ' 男 总 是 为 真 。 
(3) 保存 文件 ,上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 ( 参 见 图 3-1 .图 3-2 和 
图 3-3)。 
在 此 例 中 使 用 了 条 件 语 句 的 嵌 套 ,把 一 个 条 件 语 句 放 在 另 一 个 条 件 语句 内 部 。 遇 到 
这 种 复杂 语句 结构 时 ,可 以 使 用 制 表 位 (或 4 个 空格 ) 实 现代 码 的 缩 进 ,以 这 种 方式 格式 化 
条 件 语句 是 一 个 标准 的 过 程 和 良好 的 编程 形式 。 
这 个 脚本 中 的 第 一 个 条 件 语 句 (isset()) 是 一 个 检查 默认 值 的 理想 示例 。 在 后 续 学 习 
了 HTML 表单 后 ,可 使 用 PHP 的 表单 变量 来 获取 用 户 实际 选择 的 性 别 ( $ gender) 变量 
的 值 。 
另外 ,如 果 只 执行 一 条 语句 , 则 用 于 指示 条 件 语句 开始 和 结束 的 花 括 号 不 是 必需 的 。 
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请 您 重新 输入 您 的 性 别 。 


图 3-3 如 果 没 有 选择 性 别 (未 初始 化 $ gender) ,就 会 打印 一 条 消息 ,指出 用 户 的 疏忽 
不 过 ,出 于 对 清晰 性 的 考虑 ,在 这 里 建议 初学 者 使 用 它们 。 
3.1.2 ”switch 条 件 语句 


PHP 还 有 另 一 种 条 件 语 句 , 称 为 switch, 能 够 最 佳 地 用 于 代替 较 长 的 if-elseif-else 条 
件 语 名 。 上 一 节 介绍 的 话语 句 只 有 两 个 分 支 可 供 选择 ,而 实际 问题 中 常 需要 用 到 多 分 支 
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的 选择 ,比如 ,成 绩 的 等 级 分 为 优秀 良好、 及格 或 不 及 格 等 。 当 然 可 以 通过 使 用 多 个 if 
语句 ,或 骨 套 的 计 语 句 来 处 理 , 但 是 如 果 分 支 较 多 ,就 会 使 程序 宛 长 而 且 可 读 性 较 差 。 
PHP 提供 了 分 支 (switch) 语 名 来 直接 处 理 多 分 支 选 择 。switch 的 语法 是 : 


switch($ variable) { 
case ' 值 1': 
语句 1 
break; 
case ' 值 2': 
语句 2 
break; 
default: 
其 他 语句 
break; 
} 
switch 语句 是 一 行 接 一 行 地 执行 的 ,开始 时 没有 代码 被 执行 。switch 条 件 语 句 将 
$ variable 变量 的 值 与 不 同 的 case 作 比 较 。 仅 当 一 个 case 语句 中 的 值 和 $ variable 变量 
的 值 匹配 时 PHP 才 开始 执行 语句 ,直到 switch 的 程序 段 结束 或 者 遇 到 第 一 个 break 语 
句 为止。 如 果 不 在 case 的 语句 段 最 后 写 上 break,PHP 将 继续 执行 下 一 个 case 中 的 语 
句 段 。 一 个 case 的 特例 是 default。 它 匹配 了 任何 与 其 他 case 都 不 匹配 的 情况 ,并 且 应 
该 是 最 后 一 条 case 语句 。 如 果 没 有 发 现 匹 配 , 则 会 执行 default, 它 是 可 选 的 。 
因此 ,脚本 3-1 中 的 条 件 语句 可 以 重 写 为 ; 


switch($ gender) { 

case ' 男 ': 
echo '<p><b> 您 好 ,先生 !< /b>< /p>"'; 
break; 

case ' 女 ': 
echo '<p><b> 您 好 ,女士 !< /b></p>'; 
break; 

default: 
echo '<p><b> 请 您 重新 输入 您 的 性 别 。< /b>< /p> '; 
break; 


1 
需要 注意 的 是 ,switch 条 件 语 句 仅 限于 在 可 以 检查 变量 值 与 某 些 情况 的 相等 性 的 条 
件 下 使 用 ,往往 不 能 轻松 地 检查 更 复杂 的 条 件 语句 。 
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3.2 再 环 结 构 


下 面 将 讨论 的 语言 构造 是 循环 ,有 两 种 常用 的 循环 类 型 : while 和 for。 
3.2.1 while 循环 


1. while 循环 
while 循环 是 PHP 中 最 简单 的 循环 类 型 。while 语句 的 基本 格式 是 : 


while (条 件 ) 
语句 

只 要 判断 条 件 为 TRUE, 就 重复 执行 府 套 中 的 循环 语句 。 表 达 式 的 值 在 每 次 开始 循 
环 时 检查 ,所 以 即使 这 个 值 在 循环 语句 中 改变 了 ,语句 也 不 会 停止 执行 ,直到 本 次 循环 结 
束 。 如 果 表 达 式 的 值 一 开始 就 是 FALSE, 则 循环 语句 一 次 都 不 会 执行 。 后 面 在 第 9 章 中 
将 介绍 , 当 从 数据 库 中 检索 结果 时 ,经常 使 用 while 循环 。 

和 让 语句 一 样 ,可 以 在 while 循环 中 用 花 括号 括 起 一 个 语句 组 。 例 如 : 

$i=1; 

while ($i <=10){ 

echo $i++? // 从 1~10 依 次 输出 
} 


2. do-while 循环 
do-while 和 while 循环 非常 相似 ,区 别 在 于 表达 式 的 值 是 在 每 次 循环 结束 时 检查 而 
不 是 开始 时 。do-while 语句 的 基本 格式 是 : 
gof 
语句 
jwhile (条 件 ) 
和 while 循环 主要 的 区 别 是 do-while 的 循环 语句 保证 会 执行 一 次 ,因为 do-while 循 
环 的 表达 式 是 否 为 真 在 每 次 循环 结束 后 检查 .然而 在 while 循环 中 就 不 一 定 了 ,因为 
while 循环 的 条 件 是 否 为 真 在 循环 开始 时 检查 ,如 果 一 开始 就 为 False, 则 整个 循环 立即 终 
止 。 例 如 : 
$i=0; 
do { 
echo $i; 
}while($i >0); 
以 上 循环 将 正好 运行 一 次 ,因为 经 过 第 一 次 循环 后 , 当 检 查 表达 式 是 否 为 真 时 ,其 值 
为 False( $i 不 大 于 0) 而 导致 循环 终止 。 
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3. 使 用 while 和 do-while 循环 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 3-2) 。 


脚本 3-2 使 用 while 和 do-while 循环 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DID XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll1- transitional .dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv= "Content- Type" content="text/html; charset=gb2312" /> 
6 <title>while 和 do- while 循环 < /title> 

7 </head> 

8 <body> 

bs 


10 <body> 

11 <?php 

12 $num0; 

13 while($num<10){ 

14 echo "第 ".$num." 次 循环 。<br />"; 
15 $numt + 


22 echo "第 ".$1." 次 循环 。<br />"; 
23 Si++1 

24 J}while($i<10); 

25 ?> 

26 </body> 

27 </htm> 


(2) 使 用 while 循环 实现 10 次 循环 。 


$ num=0; 
while($num<10){ 
echo "第 ".5num." 次 循环 。<br />"; 
Snumt+? 
} 


(3) 使 用 do-while 循环 实现 10 循环 。 


$i=0; 

dof 
echo "第 ".$i." 次 循环 <br /> "7 
Si++7 

J}while ($i<10); 
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两 次 循环 设 定 循环 变量 分 别 是 $ num 和 $i, 两 次 循环 的 显示 效果 是 一 致 的 ,都 是 从 
0 开始 循环 至 9, 共 10 次。 

(4) 保存 文件 ,上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 ( 参 见 图 3-4) 。 
||@ wicinao wi >| 


全 rapyocahosvbcP - © x 


图 3-4 while 和 do-while 循环 


3.2.2 for 循环 


1. for 循环 
for 循环 具有 更 复杂 的 语法 : 


for (exprl; expr2; expr3){ 
statement 


} 


在 第 一 次 执行 循环 时 ,会 运行 初始 表达 式 (exprl) ,然后 检查 条 件 (expr2) ,如 果 条 件 
为 真 ,就 执行 循环 的 内 容 (statement)。 执 行 之 后 ,将 会 运行 结束 表达 式 (expr3) ,并 再 次 
检查 条 件 (expr2)。 这 个 过 程 会 继续 下 去 ,直到 条 件 为 假 。 例 如 : 


for($i=1;$ i<=10;$ i++){ 
echo $i; 


} 


第 一 次 运行 这 个 循环 时 ,将 把 $i 变量 的 初始 值 设置 为 1, 然 后 检查 条 件 (1 小 于 或 等 
于 10 吗 ?) ,由 于 这 个 条 件 为 真 ,就 会 打印 出 1(echo $1); 然 后 将 $i 递增 为 2( $i 十 十 )， 
再 检查 条 件 , 以 此 类 推 。 这 个 脚本 的 执行 结果 将 打印 出 数字 1 一 10。 

在 这 里 需要 注意 的 是 ,由 于 for 和 while 循环 的 语法 和 功能 足够 相似 ,因此 经 常 可 以 
互 换 使 用 它们 。 尽 管 如 此 ,经 验 表明 : 当 要 把 某 件 事情 做 一 定 的 次 数 时 ,for 循环 是 更 好 
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的 选择 ;只 要 条 件 为 真 就 做 某 件 事情 时 ,最 好 使 用 while 循环 。 
使 用 循环 时 ,要 注意 观察 参数 和 条 件 , 避 免 可 怕 的 无 限 循环 , 当 循环 条 件 永远 不 会 为 
假 时 ,就 会 发 生 这 种 情况 。 


2. 


使 用 for 循环 


(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 3-3) 。 


脚本 3-3 使 用 for 循环 。 


< !DOCTYPE html PUBLIC "- //W3C//DITD XHTML 1.0 Transitional//EN” 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtmll1- transitional .dtd"> 
<html xmlns= "http://www.w3.0rg/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title> for 循 环 < /title> 
</head> 
<body> 


<body> 
<?php 
for($i=0;$i<10;$i++){ 
echo "第 ".$1." 次 循环 。< /br>"; 
3 
?> 
< /body> 
</html> 


(2) 使 用 for 循环 实现 10 次 循环 。 


for($i=0;$ i<10;$i++){ 


} 


echo "第 ".5i." 次 循环 。< /br>"; 


循环 变量 为 $i, 循 环 次 数 从 0 一 9,. 共 10 次 。 
(3) 保存 文件 ,上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 (参见 图 3-5) 。 


3-5 for 循环 
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3.3 项 目 训练 


动态 年 月 日 下 拉 菜 单 


3.3.1 项 目 说 明 


网 络 上 众多 注册 表单 在 涉及 选择 日 期 或 生日 等 表单 控件 时 ,多 数 会 使 用 下 拉 列 表 。 
通过 PHP 循环 ,实现 下 拉 列 表 年 月 日 自动 生成 为 列表 菜单 项 目 。 


3.3.2 设计 思路 


不 管 是 年 月 日 的 哪 一 项 ,都 是 一 定 范围 内 的 整数 集合 ,年 份 可 以 是 2005 一 2015 年 ,月 
份 规定 为 1 一 12 月 ,日 期 为 1 一 31 日 ,这 里 暂且 忽略 月 份 之 间 天 数 的 差异 。 

为 此 ,同样 还 是 先 制作 静态 页 面 , 制 作 一 个 包含 3 个 下 拉 菜 单 格式 的 Web 页 ,然后 使 
用 任何 一 种 PHP 的 循环 结构 去 实现 有 规律 的 整数 范围 ,并 且 年 份 的 周期 也 方便 在 日 后 
随时 修改 。 


3.3.3 设计 过 程 


循环 实现 动态 年 月 日 下 拉 菜 单 , 这 里 分 别 使 用 了 while 循环 .do-while 循环 及 for 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 3-4) 。 


脚本 3-4 ”使 用 循环 实现 下 拉 菜 单 。 


1 <“ !DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll1- transitional.dtd"> 

3 “html xmlns= "http://www.w3.org/1999/xhtml"> 

4 “head> 

5 <meta http- equiv= "Content- TYPpe”content= "text/html; charset=gb2312"”/> 
6 ”<title>while 和 for 循 环 结构 < /title> 

7 </head> 

8 <body> 

9 


10 <form action="" method= "post"> 

11 <?php # 脚 本 3- 4- calendar.php 

12 // 创 建 年 份 下 拉 菜单 

13 echo '< select name= "year"> '; 

14 $year=2005; 

15 while($year <=2015){ 

16 echo "< option value= \"$ year\"> $ year< /option> \n"; 
7 $yeart+?} 

18 } 

19 echo '</select> 年 '; 
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21 // 创 建 月 份 下 拉 菜 单 

22 echo '<select name= "month">'; 

23 $month=1; 

24 do{ 

25 echo "< option value= \"$ month\"> $ month< /option> \n"} 
26 $montht +} 

27 J}while($month <=12); 

28 echo '</select> 有 月"'; 


30 // 创 建 日 期 下 拉 菜 单 
echo '< select name= "day"> '7 
for($day=1; $day <=31; $day+t+){ 
echo "< option value= \"$ day\"> $ day< /option> \n"; 
} 
echo '< /select> 日 ';» 


出 呈 央 总 是 


36 
37 ?> 

38 </formm> 
39 </body> 
40 </html> 


这 里 需要 注意 的 是 ,需要 HTML 表单 标签 创建 下 拉 菜 单 。 因 为 在 何 处 以 及 何 时 髓 


入 PHP 是 无 关 紧 要 的 ,所 以 这 里 把 表单 标签 放 在 PHP 代码 前 面 。 
(2) 使 用 while 循环 生成 年 份 下 拉 菜 单 。 


echo "< select name= "Year"> '; 

$ year= 2005; 

while($ year <=2015){ 
echo "< option value=\"$ year\"> $ year< /option> \n"; 
$yeart+? 

} 

echo '< /select> 年 '; 


最 初 设置 $ year 变量 ,只 要 它 小 于 或 等 于 2015 ,就 会 执行 循环 。 在 循环 内 ， 


echo() 语 句 , 然 后 递增 $ year 变量 。 
(3) 使 用 do-while 循环 生成 月 份 下 拉 某 单 。 


echo '< select name= "month"> '; 

$month=1; 

do{ 
echo "<option value=\"$ month\"> Smonth< /option> \n"7 
Smonth++; 

J}while ($month <=12); 

echo '</select> 月 '; 


这 里 使 用 的 do-while 循环 生成 月 份 与 while 循环 生成 年 份 功能 相似 。 
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(4) 使 用 for 循环 生成 日 期 下 拉 菜 单 。 


echo "< select name= "day"> '7 
for($day=1; $day <=31; $day++){ 
echo "<option value=\"$ day\"> $ day< /option> \n"7 
} 
echo '</select> 日 '; 


这 个 标准 的 for 循环 首先 将 $ day 变量 初始 化 为 1。 它 将 继续 运行 循环 ,直到 $ day 
大 于 31, 并 在 每 次 迭代 时 ,将 $ day 递增 1。 循 环 自身 的 内 容 ( 它 将 执行 31 次 ) 是 一 条 
echo() 语 句 。 


(5) 将 文件 另存 为 calendar. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 ( 参 
见 图 3-6) 。 


3-6 ”这些 下 拉 菜 单 是 使 用 循环 结构 创建 的 


(6) 选择 “查看 ”>“ 源 文件 "命令 ,或 “查看 ”>“ 页 面 源 文件 "命令 ,在 浏览 器 中 查看 源 
代码 (参见 图 3-7) 。 


i< 
了 "http://wew.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
3 <html xmlns="http://www.w3.org/1999/xhtml” xml:lang="en” lang="en"> 
4 <head> 


5 <meta http-equiv="content-type"” content="text/html; charset=gh2312" /> 
6 <title> 年 历 </title> 

7 </head> 

8 <body> 

3 

10 <form action="" method="post"> 

11 <select name="year"><option value="2005">2005</option> 

12 <option value="2006">2006</option> 

13 <option value="2007">2007</option> 

14 <option value="2008">2008</option> 

15 <option Value="2009">2009</option> 

16 <option value="2010">2010</option> 

17 <option value="2011">2011</option> 

18 <option value="2012">2012</option> 

19 <option value="2013">2013</option> 

20 <option value="2014">2014</option> 

21 <option value="2015">2015</option> 

lect name="month"><option value="1">1</option> 
">2</option> 


25 <option value: 
26 <option value="5">5</option> 


3-7 大 多 数 源 代码 都 是 由 仅仅 几 行 PHP 代码 生成 的 
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本 章 小 结 


本 章 介绍 了 PHP 语言 的 条 件 语 句 及 其 余 的 运算 符 , 以 及 循环 语言 构造 。 这 些 语言 
基础 将 会 为 完成 后 续 的 复杂 程序 设计 提高 效率 。 


重点 回顾 
1. PHP 的 比较 和 逻辑 运算 符 。 


2. 如 何 使 用 分 支 语句 让 结构 与 switch 结构 实现 各 种 条 件 判断 。 
3. 合理 使 用 循环 结构 实现 高 效率 的 编程 。 


本 章 实 训 
【 实 训 1 


新 建 switch. php, 参 考 PHP 中 文 帮助 ,使 用 date 函数 获取 今天 星期 几 , 然 后 使 用 
switch 分 支 语句 分 别 给 出 不 同 说 明 。 网 页 运行 效果 如 图 3-8 所 示 。 


Cy 


今天 星期 一 ， 猴子 穿 新 衣 。 


图 3-8 switch. php 的 运行 效果 


【 实 训 2】 
新 建 99. php, 使 用 for 循环 实现 九 九 乘法 表 . 并 将 结果 输出 为 表格 格式 。 网 页 运行 效 
果 如 图 3-9 所 示 。 


#1=1 [ly2=2 [i#3=3 li*d=4 [i#5=5 [1#6=6 [I#7=7 [i#8=8 [I*9=9 
[2#*1=2 (2+2=4 [2+3=6 [2+4=8 |2+5=10 |2+6=12 [2*7=14 |2+8=16 |2+9=18 
B+1=3 [Br2=6 [3*3=9 [34=12 [3*5=15 [3+6=18 [3#7=21 [3+8=24 [3*9=27 
(#1=4 (28 [312 (4*d=16 |4#5=20 |4+6=24 [4*7=28 [4+8=32 (4*9=36 
(5+1=5 [5#2=10 [5*3=15 |5#*4=20 [5#5=25 |5*6=30 [5#7=35 [5#8=40 |5+9=45 
G+*1=6 [6*2=12 [6*3=18 [6*4=24 [6*5=30 |646=36 [6*7=42 |6+8=48 [6*9=54 
[#1=7 [7#2=14 [7+3=21 [7*4=28 |7+5=35 [7#6=42 T+8=56 (7#9=63 


[B*1=8 [8+2=16 |8+3=24 [8+4=32 [8+5=40 [8+6=48 B+8=64 |8*9=72 
[9+1=9 (9*2=18 [9+3=27 |9+4=36 |9+5=45 |9+6=54 [9#7=63 |9+8=72 [3*9=81 


图 3-9 ”99. php 的 运行 效果 
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数组 是 一 种 数据 类 型 , 它 的 使 用 频率 相当 高 .学 会 处 理 数 组 会 使 我 们 在 编程 时 得 心 应 
手 。 数 组 同 数值 或 者 字符 串 有 着 显著 的 不 同 ,并 且 如 果 没 有 掌握 和 理解 它 , 大 多 数 PHP 
编程 工作 都 将 无 法 实现 。 


4.1 什么 是 数组 


数组 (array) 组 成 一 个 复杂 但 是 非常 有 用 的 概念 。 数 值 和 字符 串 都 是 标量 变量 ,它们 
只 含有 一 个 单独 的 值 , 数 组 实际 上 是 一 个 数据 集合 , 它 可 以 保存 多 份 单独 的 信息 ,就 像 是 
值 的 列表 ,其 中 每 个 值 可 以 是 字符 串 或 数字 ,甚至 是 另 一 个 数组 。 因 此 比 起 简单 的 字符 串 
或 者 数字 ,数组 能 够 让 一 个 变量 以 指数 级 承载 更 多 的 信息 。 


4.1.1 索引 数组 与 联合 数组 


数组 就 是 一 组 数据 的 集合 ,把 一 系列 数据 组 织 起 来 ,形成 一 个 可 操作 的 整体 。 数 组 的 
每 个 实体 都 包含 两 项 : 键 和 值 。 
数组 可 以 构造 成 一 系列 键 - 值 (key-value) 对 ,其 中 每 一 对 都 是 那个 数组 的 一 个 项 目 或 
元 素 (element)。 对 于 列表 中 的 每 个 项 目 ,都 有 一 个 与 之 关联 的 键 (key) (或 索引 
(index))。 得 到 的 结构 很 像 是 电子 表格 或 数据 库 表 (参见 表 4-1 和 表 4-2) 。 
表 4-1 $days 索引 数组 使 用 数字 作为 它 的 键 


键 值 键 值 

0 星期 天 3 星期 三 

1 星期 一 4 星期 四 

2 星期 二 

表 4-2 $user 联合 数组 使 用 字符 串 作为 它 的 键 

键 值 键 值 
id 0922188 gender 男 
name 张 明 age 30 
pwd 1234567890 
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PHP 支持 两 种 数组 : 索引 数组 (indexed array) 和 联合 数组 (associative array) ,前 者 
使 用 数字 作为 键 ,后 者 使 用 字符 串 作 为 键 。 

同 C 语言 和 大 部 分 语言 一 样 , 索 引 数 组 的 第 一 个 索引 开始 于 0, 而 不 是 从 1。 键 (key) 
可 以 是 整 型 或 者 字符 串 , 值 (value) 可 以 是 任何 类 型 。 例 如 , $ days 数组 ,如 表 4-1 所 示 。 

联合 数组 就 是 在 索引 值 方 面 与 索引 数组 不 同 , 其 索引 值 中 包含 了 字符 串 ,这 样 可 以 更 
方便 地 处 理 实际 生活 的 问题 ,比如 把 名 字 作为 索引 ,资料 为 索引 的 值 ,但 是 在 操作 数据 时 
就 需要 比较 复杂 的 方法 。 例 如 , $ user 数组 ,如 表 4-2 所 示 。 

数组 遵守 与 任何 其 他 变量 相同 的 命名 规则 .数组 的 名 称 使 用 美元 符号 ($ ) 开 头 。 因 
此 , 单 看 变量 名 $ var 无 法 区 分 是 数组 ,还 是 字符 串 或 数字 。 

由 于 PHP 对 其 变量 结构 的 要 求 很 宽松 ,数组 甚至 可 以 使 用 数字 和 字符 串 的 组 合作 
为 它 的 键 。 唯 一 重要 的 规则 是 数组 的 每 一 个 键 都 必须 是 唯一 的 。 


4.1.2 创建 数组 


创建 一 个 数组 是 指定 义 一 个 数组 的 名 字 和 结构 ,可 以 初始 化 其 内 部 数据 元 素 的 值 ,也 可 
以 不 作 初 始 化 处 理 , 即 所 有 的 元 素 值 为 空 。 在 PHP 中 ,创建 或 声明 数据 的 方式 主要 有 两 种 : 
一 种 是 直接 为 数组 元 素 赋值 ,一 次 添加 一 个 元 素来 构建 数组 ; 另 一 种 是 应 用 array() 函数 声 
明 数 组 。 

1. 直接 为 数组 元 素 赋值 

直接 为 数组 元 素 赋值 方法 是 使 用 方 括号 的 语法 ,通过 在 方 括号 内 指定 键 为 数组 赋值 
来 实现 ,也 可 以 省 略 键 ,在 这 种 情况 下 给 变量 名 加 上 一 对 空 的 方 括号 ([])。 语 法 如 下 : 


$arrayName [key]=value; 


$arrayName []=value; 
其 中 , 键 key 可 以 是 整 型 或 者 字符 串 , 值 value 可 以 是 任何 类 型 。 例 如 : 


$colName [0]= "id"; 
$ colName [1]= "username"; 
$ colName [2]= "pwd"; 


如 果 给 出 方 括号 但 没有 指定 键 , 则 取 当 前 最 大 整数 索引 值 , 新 的 键 将 是 该 值 加 1。 如 
果 当 前 还 没有 整数 索引 , 则 键 将 为 0。 例如: 


$array[]= 'Math'; 
Sarray[]= "9 月 1 日 


$array['teacher']='Chen'; 

需要 注意 的 是 ,如 果 指 定 一 个 键 ,并 且 已 经 存在 用 那个 相同 的 键 进行 索引 的 一 个 值 ， 
则 新 值 将 覆盖 现 有 的 值 。 例 如 : 

$array['son']= "Mark'; 

$array['son']= "Michael'; 

$array[2]= 'apple'; 
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$array[2]= 'orange'7 

如 果 在 创建 数组 时 不 知 所 创建 数组 的 大 小 ,或 在 实际 编写 程序 时 数组 的 大 小 可 能 发 
生变 化 ,那么 采用 这 种 数组 创建 的 方法 较 好 。 

2. 使 用 array() 函 数 创建 数组 

创建 数组 正式 的 方法 是 使 用 array() 函 数 。 它 的 语法 如 下 : 

$1list=array('apple'，"banana'，'orange'")7 

除非 特别 指出 ,否则 数组 的 索引 自动 从 0 开始 。 本 示例 并 未 为 元 素 指定 特定 的 索引 ， 
因此 apple 作为 第 一 项 自动 索引 为 0, 而 第 二 项 索引 为 1, 第 三 项 索引 为 2。 

可 以 在 使 用 array() 时 为 索引 赋值 ,运算 符 * 王 二 ”用 于 定义 数组 元 素 的 值 。 语 法 “key 
三 过 values”, 用 于 定义 数组 键 和 对 应 的 值 。 键 可 以 是 数字 或 字符 串 。 例 如 : 

$1ist=array (l=> 'Monday', 2=> 'Tuesday', 3=> 'Wednesday') 

$fruit=array ("first"=> 'apple',"second"=> 'banana', "third"=> 'orange'); 

最 后 ,指定 的 索引 值 不 必 为 数值 ,还 可 以 使 用 字符 串 。 这 种 索引 的 技术 在 制作 更 加 有 
意义 的 列表 时 非常 实用 。 


4.1.3 数组 的 打印 


1. 打印 单个 数组 元 素 

在 PHP 中 对 数组 元 素 输出 ,可 以 通过 echo 和 print 语句 来 实现 ,但 这 只 能 对 数组 中 
的 单个 元 素 进 行 输出 。 要 从 数组 中 检索 特定 的 值 ,需要 先 引 用 数组 名 称 , 然 后 在 方 括号 中 
指出 键 。 例 如 : 

echo $days [2]; // 输 出 星期 二 

echo $user ['name']; // 输 出 张 明 
可 以 看 到 在 PHP 中 , 若 数组 的 键 值 为 数字 (例如 ,2) 则 不 会 用 引号 括 住 , 而 对 字符 串 ( 例 
如 ,name) 则 必须 这 样 做 。 

由 于 数组 使 用 的 语法 不 同 于 其 他 变量 ,对 它们 执行 打印 更 具 技巧 性 ,以 下 是 容易 犯 的 
错误 。 
(1) 因为 数组 可 以 包含 多 个 值 ,所 以 不 能 编写 简单 的 代码 : 
echo "输出 星期 : $ day"; 
(2) 联合 数组 中 的 键 复杂 化 了 打印 它们 的 操作 。 下 面 的 代码 将 引起 一 个 解析 错误 : 
echo "客户 的 姓名 是 $user['name']"; 
为 了 解决 这 个 问题 , 当 数 组 使 用 字符 串 作为 它 的 键 时 .把 数组 名 和 键 包装 在 花 括 
号 中 : 

echo "客户 的 姓名 是 {$user["'name']}"; 
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不 过 ,数字 式 索 引 数 组 没有 这 种 问题 : 
echo "今天 是 $days [4]"; 


2. 输出 整个 数组 结构 
要 将 数组 结构 输出 则 要 使 用 print_r() 函 数 。 语 法 如 下 : 


print_r (mixed expression); 


如 果 参 数 expression 为 普通 的 整 型 字符 型 或 实 型 变量 , 则 输出 该 变量 本 身 ;如 果 该 
参数 为 数组 , 则 按 一 定 键 值 和 元 素 的 顺序 显示 出 该 数组 中 的 所 有 元 素 。 

作为 示例 ,可 以 利用 下 面 的 脚本 创建 一 个 数组 来 记录 一 周 中 每 天 的 天 气 ,并 且 可 以 通 
过 使 用 赋值 运算 符 为 每 个 元 素 赋值 的 方式 直接 向 数组 添加 元 素 。 该 示例 也 将 诠释 如 何 能 
够 以 及 不 能 打印 数组 。 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 文档 (参见 脚本 4-1): 


脚本 4-1 。 $$ weather 数组 包含 3 个 元 素 并 且 使 用 字符 串 作 为 它 的 键 。 


1 <!DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtmll/DTD/xhtmll1- transitional.dtd"> 

3 “html xmlns= "http://www.w3.org/1999/xhtml"> 

4 “head> 

5 <meta http-equiv= "Content- Type" content="text/html; charset=gb2312”/> 
6 ”<title> 打 印 数组 < /title> 

7 </head> 

8 <body> 

9 


10 <?php //Script 4-1-weatherl.php 
11 // 创 建 数 组 : 

12 $weather=array( 

13 ”'Monday'=> ' 晴 转 多 云 '， 

14 "Tuesday'=>' 多 云 转 阴 '， 

15 "Wednesday'= > ' 阴 转 阵 雨 ' 

16 'Thursday'=> ' 多 云 ' 

ET 


19 // 打 印 数组 结构 及 内 容 : 
20 print r($weather); 


22 // 使 用 print 打印 数组 : 
23 print "<p>$ weather< /p>"; 


25 // 为 数组 再 添加 3 个 元 素 : 
26 $weather['Friday']=' 多 云 转 睛 '; 
27 $weather['Saturday']=' 多 云 转 阴 '; 
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28 $weather['Sunday']=' 阴 有 阵雨 '; 


30 var_dump ($ weather); 
对 1 

32 </body> 

33 </htm> 


(2) 开始 脚本 中 的 PHP 片段 ,使 用 array() 函数 创 建 一 个 数组 : 


$ weather=array( 
"Monday'=> ' 晴 转 多云 '， 
"Tuesday'=>' 多 云 转 阴 '， 
"Wednesday'=> ' 阴 转 阵雨 '， 
"Thursday'=> ' 多 云 ' 
) 7 
这 是 在 PHP 中 用 字符 串 作 为 索引 对 数组 进行 初始 化 (创建 和 为 其 赋值 ) 所 使 用 的 正 
确 格式 。 由 于 其 键 和 值 都 是 字符 串 类 型 ,因此 需要 用 单 引 号 将 之 引用 。 
不 要 在 最 后 一 个 数组 元 素 之 后 添加 逗号 (索引 为 Thursday) ,这 样 做 会 导致 解析 错误 
发 生 。 
(3) 尝试 打印 这 个 数组 : 


Print r($weather); 


使 用 print_r() 函 数 可 以 来 显示 任何 变量 的 内 容 和 结构 。 
(4) 使 用 与 前 述 不 同 的 printO 〇 函数 打印 数组 : 


Print "<p> $weather < /p>"; 


如 图 4-1 所 示 , 这 里 将 不 会 显示 数组 内 容 , 由 于 数组 和 其 他 变量 类 型 是 结构 上 不 同 
的 ,因此 直接 使 用 “print” 方 法 打印 数组 的 请 求 会 导致 数值 类 型 名 称 *Array” 的 出 现 (以 及 
错误 的 出 现 ,这 取决 于 PHP 的 版 本 及 其 设置 )。 

(5) 为 数组 再 添加 3 个 元 素 : 

$weather['Friday']= ' 多 云 转 睛 '; 

$weather['Saturday']= ' 多 云 转 阴 '; 

S$weather['Sunday']= ' 阴 有 阵雨 '; 

这 些 代码 又 向 已 有 的 数组 添加 了 3 天 的 天 气 . 分 别 用 Friday、Saturday 和 Sunday 进 
行 索引 。 

(6) 再 次 打印 数组 。 

这 里 用 var_dump() 函数 代替 print_r() , 它 不 仅 显 示 数 组 的 内 容 . 同 时 还 会 以 更 加 详 
细 的 格式 呈现 数组 的 结构 ,如 图 4-1 所 示 。 在 PHP 第 6 版 中 ,这 个 函数 还 会 指出 一 个 字 
符 串 是 否 使 用 了 Unicode 编码 。 

(7) 结束 PHP 和 HTML 代码 部 分 ,将 文档 保存 为 weatherl. php ,放置 在 启用 了 
PHP 的 服务 器 上 适当 的 目录 中 ,并 且 在 Web 浏览 器 中 进行 测试 (参见 图 41) 。 
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Array ( [onday] => 晴 转 多 云 [Tuesday] = 多 云 转 阴 
[Wednesday] => 阴 有 阵雨 [Thursday] =》 多 云 ) 


Array 


array(7) { [onday”]=> string(8)“ 晴 转 多 

云 ”[“Tuesday”]=》 string(8) “多云 转 阴 ”[“Wednesday”]=> 
string(8)“ 阴 有 阵雨 ”[“Thursday”]=> string(4)“: 

云 ”["Priday”]=》string(8)“ 多 云 转 晴 ”[“Saturday”]=> 
ss “多 云 转 阴 ”[”Sunday”]=》string (8)“ 阴 有 阵 


图 4-1 weatherl. php 的 执行 结果 


4.2 访问 数组 


4.2.1 foreach 循环 


无 论 是 如 何 创建 数组 的 ,从 数组 中 检索 某 个 特定 元 素 ( 或 者 值 ) 只 有 一 个 方法 , 那 就 是 
用 它 的 索引 。 要 访问 数组 中 所 有 值 的 最 快捷 而 且 最 简单 的 方法 是 使 用 foreach 循环 。 这 
个 循环 结构 将 遍历 数组 中 的 每 个 元 素 。 
要 访问 每 个 数组 元 素 , 可 以 使 用 foreach 循环 。foreach 循环 有 两 种 格式 。 如 果 只 访 
问 数组 元 素 的 值 ,可 以 使 用 : 
foreach ($array as $value){ 
echo $value; //$value 变量 名 字 可 以 随意 合法 设置 
} 
foreach 循环 将 会 迭代 $ array 中 的 每 个 元 素 . 并 把 每 个 元 素 的 值 赋予 $ value 变量 。 
如 果 要 访问 键 和 值 ,可 以 使 用 : 
foreach ($array as $ key= > $value){ 
echo " 元 素 的 键 为 :$key ,对 应 的 值 为 :$value."; //$key 和 $value 变 量 名 字 可 以 随意 合法 设置 
} 
现在 可 以 利用 这 些 知 识 编 写 一 个 新 的 weather 脚本 。 不 仅仅 能 够 打印 出 数组 中 拥有 
的 元 素数 量 , 还 将 能 够 访问 到 它 真实 的 值 。 
1. 打印 数组 的 值 
(1) 在 文本 编辑 器 中 新 建 一 个 文档 (参见 脚本 4-2) 。 


脚本 4-2 ”foreach 循环 是 访问 数组 中 每 个 元 素 的 最 简便 的 方法 。 


1 < !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN™” 


"http://www-w3.org/TR/xhtml1/DTD/zhtml1-transitional.dtd"> 


<html xmlns= "http://www.w3.0rg/1999/xhtml"> 
<head> 


第 4 章 数 组 


<meta http- equiv= "Content- Type”" content= "text/html; charset=gb2312" /> 


- 
3 
4 
5 
6 ”<title> 打 印 数组 < /title> 
7 </head> 

8 <body> 

9 

10 <?php //Script 4- 2- weather2.php 
11 。// 创 建 数组 : 


12 S$weather=array( 


13 'Monday'=> ' 晴 转 多 云 '， 

14 'Tuesday'=> ' 多 云 转 阴 '， 
15 "Wednesday'=> ' 阴 有 阵雨 '， 
16 "Thursday'=> ' 多 云 '， 

17 'Friday'=> ' 多 云 转 晴 '， 
18 "Saturday'=> ' 多 云 转 阴 '， 
19 'sunday'=> ' 阴 有 阵雨 ' 

20 ); 

21 


22 // 打 印 数组 的 值 : 

23 foreach($ weather as $day=>$ content){ 
24 echo "<p> $day: $ content< /p> \n"; 
25 } 

26 

27 ?> 

28 </body> 

29 </html> 


(2) 开始 脚本 中 的 PHP 片段 ,使 用 array() 函 数 创建 一 个 数组 : 


$ weather=array ( 
"Monday'=> ' 晴 转 多 云 '， 
"Tuesday'=> ' 多 云 转 阴 '， 
'Wednesday'=>' 阴 有 阵雨 '， 
'Thursday'=> ' 多 云 '， 
'Friday'=> ' 多 云 转 晴 '， 
'Saturday'=> ' 多 云 转 阴 '， 
'sunday'=> ' 阴 有 阵雨 ' 

); 


在 这 里 一 次 就 创建 了 整个 数组 ,虽然 可 以 使 用 同 前 面 脚 本 中 相同 的 方式 ,逐步 创建 


数组 。 


(3) 创建 一 个 foreach 循环 以 打印 每 天 的 天 气 : 


foreach ($ weather as $ day=> $ content){ 
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echo "<p> $day: 5 content< /p> \n™; 

} 

foreach 循环 遍历 数组 $ weather 的 每 个 元 素 , 将 每 个 索引 赋值 给 $ day, 并 将 每 个 值 
赋值 给 $ content。 这 些 值 现在 打印 在 HTML 段落 标记 中 。 该 echo 语句 包含 一 个 换行 
符 ( 使 用 \n 创建 的 ) , 它 将 对 页 面 中 的 HTML 源 代码 造成 影响 。 

(4) 结束 PHP 部 分 和 HTML 页 面 : 

和 

< /body> 

</html> 

(5) 将 页 面 保存 为 weather2. php, 并 放置 在 启用 了 PHP 的 服务 器 上 适当 的 目录 中 ， 
然后 在 Web 浏览 器 中 进行 测试 (参见 图 4-2) 。 


(HD @ htpi//localhosbe PD - © x | @ imma 
Jonday: 晴 转 多 云 


| Tuesday: 多 云 转 阴 
Wednesday: 阴 有 阵雨 


Thursday: 多 云 
Friday: 多 云 转 晴 
Saturday: 多 云 转 阴 
Sunday: 阴 有 阵雨 


4-2 为 数组 中 的 每 个 元 素 执行 循环 后 将 生成 这 个 页 面 


2. 创建 和 访问 数组 

使 用 数组 来 建立 一 组 表单 下 拉 菜 单 以 从 中 选择 一 个 日 期 ,也 非常 容易 。 
(1) 在 文本 编辑 器 中 打开 calendar. php 脚本 (参见 脚本 3-2) 。 

(2) 添加 一 个 星期 的 下 拉 菜 单 (参见 脚本 4-3) 。 


脚本 4-3 ”循环 通常 与 数组 一 起 使 用 ,使 用 数组 动态 创建 下 拉 菜 单 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www-w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns= "http://www-w3-org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- TYpe"”content= "text/html; charset=gb2312" /> 
<title>while 和 for 循 环 结构 < /title> 
</head> 


1 
1 
3 
4 
5 
6 
| 
8 <body> 


< form action="" method= "post"> 

<?Php # 脚 本 3- 2- calendar .php 

// 创 建 年 份 下 拉 菜单 

echo '< select name= "year"> '; 

$ year= 2005; 

while ($ year <=2015) { 
echo "< option value=\"$ year\"> $ year< /option> \n"; 
$yeart+? 

} 

echo '< /select> 年 '; 


// 创 建 月 份 下 拉 菜 单 

echo '< select name= "month"> "7 

$month=1; 

dof 
echo "< option value=\"$ month\"> $ month< /option> \n"; 
$month++; 

}while ($month <=12); 

echo '</select> 月 '; 


// 创 建 日 期 下 拉 菜 单 
echo '< select name= "day"> '7 
for($day=1; $day<=31; $day++){ 
echo "< option value=\"$ day\"> $day< /option> \n"; 
} 
echo '< /select> 日 ';» 


// 创 建 星期 数组 
$weekday=array (l=> ' 星 期 一 ', ' 星 期 二 ', ' 星 期 三 ', "星期 四 ', ' 星 期 五 ', ' 星 期 六 
// 创 建 星期 下 拉 莱 单 
echo '< select name= "weekday"> '} 
foreach ($ weekday as $ key=> $value){ 
echo "< option value= \"$ key\"> $ value< /option> \n"; 
echo '< /select>'; 
2> 
</form> 
</body> 
</htm> 
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',' 星 期 天 '); 
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(3) 为 星期 创建 一 个 数组 。 
$5weekday=array (1=> "星期 一 … ' 星 期 二 ', ' 星 期 三 ', "星期 四 "星期 五 , ' 星 期 六 ', ' 星 期 天 '); 


这 个 数组 使 用 1 一 7 作为 键 。 因 为 指定 了 第 一 个 键 , 则 会 对 余下 的 值 递增 地 添加 
索引 。 
(4) 通过 foreach 循环 生成 星期 下 拉 菜 单 。 


echo '< Select name= "weekday"> '; 
foreach ($ weekday as $ key=> $value) { 
echo "< option value=\"$ key\"> $value< /option> \n"; 

} 

echo '</select>'; 

利用 foreach 循环 ,可 以 快速 生成 星期 下 拉 菜 单 的 所 有 HTML 代码 。 每 执行 一 次 循 
环 ,都 会 创建 一 行 代码 ,如 二 option value="1" 二 星期 一 一 /option 二 。 

年 月 日 的 下 拉 菜单 同样 也 可 通过 数组 的 方式 生成 ,但 是 创建 数组 会 涉及 额外 代码 和 
内 存 开销 ,所 以 使 用 循环 的 效率 更 高 。 

(5) 保存 文件 ,上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 (参见 图 4-3) 。 


4-3 这 些 下 拉 菜 单 是 使 用 循环 或 数组 创建 的 


| 


foreach 循环 仅 适用 于 数组 ,如 果 看 到 Invalid argument supplied for foreach() (为 
foreach() 提 供 了 无 效 的 参数 ) 出 错 消 息 , 则 意味 着 正 尝 试 在 不 是 数组 的 变量 上 使 用 
foreach 循环 。 


4.2.2 多 维 数组 


数组 的 值 可 以 是 数字 、 字 符 串 ,甚至 其 他 数组 的 任意 组 合 。 最 后 一 种 选择 一 一 包含 其 
他 数组 的 数组 一 一 将 会 创建 一 个 多 维 数组 。 

多 维 数组 可 以 用 来 包含 比 标准 数组 更 多 的 信息 ,多 维 数组 的 值 本 身 就 可 以 是 数组 。 
多 维 数组 的 使 用 非常 普遍 ,特别 是 在 表单 中 使 用 超 全 局 变量 时 则 更 是 如 此 。 但 是 ,处 理 多 
维 数组 非常 容易 。 例 如 : 
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$ fruits=array ('apples', 'bananas', 'oranges'); 


$ meats=array ('steaks', 'hamburgers', 'pork chops'); 


$ groceries=array( 


); 


'fruits'=>$ fruits, 
'meats'=> $meats, 
'other'=> 'peanuts', 
'cash'=>30.00 


数组 $ groceries 现在 由 一 个 字符 串 (peanuts) ,一 个 浮 点 数值 (30. 00) 和 两 个 数组 
($fruits 和 $meats) 组 成 。 

指向 多 维 数组 中 的 某 个 元 素 会 有 一 点 复杂 。 要 点 是 根据 需要 在 方 括号 中 继续 添加 索 
引 。 因 此 在 这 个 示例 中 ,bananas 位 于 $ groceries[fruits][1]。 

首先 ,使 用 [fruits] 来 指向 数组 $ groceries 中 的 元 素 ( 在 这 种 情况 下 ,是 一 个 数组 ) 。 
然后 再 基于 它 的 位 置 指向 数组 中 的 这 个 元 素 : 它 是 第 二 项 ,因此 需要 使 用 索引 [1]。 

使 用 多 维 数组 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 4-4) 。 


脚本 4-4 ”多维 数组 $interests 在 一 个 大 变量 中 存储 了 大 量 的 信息 。 


< !DOCTYPE html PUBLIC "~ //W3C/V/DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title>while 和 for 循 环 结构 < /title> 
< /head> 
<body> 


< ?php #Script 4- 4- interests.php 
// 创 建 第 一 个 数组 : 
$songs=array ("小 小 虫 ', "东风破 ', 'waka waka'); 


// 创 建 第 二 个 数组 : 
$movies=array ("钢铁 侠 ', ' 变 形 金刚 ', ' 阿 凡 达 '); 


// 创 建 第 三 个 数组 : 
$books=array(' 哈 利 波 特 ', " 暮 光 之 城 必 " 品 三 国 )7 


// 创 建 多 维 数组 : 

$ interests=array( 
‘music'=>$ songs, 
‘movie'=> $movies, 
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24 "book'=>$books 
25 ) 7 
26 // 输 出 一 些 值 : 


27 echo "<p> 我 喜欢 的 一 首 歌 是 <i> {$ interests['music'] [0]}</i>。</p>"; 
28 echo "<p> 我 喜欢 的 电影 是 <i> {$ interests['movie'] [2]}</i>. </p>"; 
29 echo "<p> 我 喜欢 的 书 是 <i> {$ interests['book'] [1]}</i>. </p>"; 


31 // 使 用 foreach 的 嵌 套 才能 完整 打印 多 维 数组 

32 foreach($ interests as $title=>$ content){ 

33 print "<p>$title"; 

34 foreach ($ content as $number=> $value) { 

35 ”print "<br /> 我 喜欢 的 第 $number 个 是 $value"; 
36 } 

37 print '</p>'; 

38 } 


40 ?> 

41 </body> 

42 </html> 

(2) 创建 初始 PHP 标签 ,创建 第 一 个 数组 : 

$5 songs=array (1=> "小 小 虫 ', "东风破 "waka waka'); 

为 了 构建 这 个 多 维 数组 ,我 们 将 创建 3 个 标准 数组 ,并 且 将 它们 作为 这 个 更 大 数组 的 
值 。 该 数组 使 用 数值 作为 键 ,同时 使 用 字符 串 作 为 值 。 数 值 从 1 开始 编号 。 

(3) 创建 接 下 来 的 两 个 数组 : 

$movies=array (1=> "钢铁 侠 ',' 变 形 金 刚 ', ' 阿 凡 达 '); 

$books=array (1=> ' 哈 利 波 特 ', ' 暮 光 之 城 ',' 品 三 国 '); 

对 于 每 个 数组 ,出 于 简便 的 考虑 ,只 添加 3 个 元 素 的 信息 。 

(4) 创建 主体 ,多 维 数组 : 


$ interests=array( 
"music'=>$ songs, 
'movie'=> $movies, 
'book'=>$ books 

); 


数组 $interests 是 本 脚本 中 的 主 数 组 。 它 使 用 字符 串 作为 键 并 且 数 组 作为 值 。 可 以 
用 函数 array() 像 创建 其 他 数组 那样 创建 它 。 
(5) 打印 一 首 歌 : 


echo "<p> 我 喜欢 的 一 首 歌 是 <i> {$ interests['music'] [1]}</i>。</p>"; 


按照 之 前 介绍 的 规则 ,为 了 访问 所 有 子 项 ,我 们 所 需要 做 的 就 是 从 $interests 开始 ， 
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首先 是 第 一 个 索引 (['music]]) ,然后 是 下 一 个 索引 ([1])。 

因为 将 把 这 些 内 容 放置 在 一 个 echo() 调 用 中 ,所 以 需要 用 花 括号 引用 整个 结构 ,以 
避免 发 生 解 析 错 误 。 

(6) 再 打印 两 个 示例 : 


echo "<p> 我 喜欢 的 电影 是 <i> {$ interests['movie'] [2]}</i>。 </p>"; 
echo "<p> 我 喜欢 的 书 是 <i> {$ interests['book'] [3]}</i>。</p>"; 


(7) 为 了 访问 每 个 数组 中 的 每 个 元 素 , 可 以 将 两 个 foreach 循环 腐 套 使 用 。 


foreach ($ interests as $title=>$ content){ 
print "<p> $title"; 
foreach ($ content as $number=> $value){ 
print "<br /> 我 喜欢 的 第 $number 个 是 $value"; 
多 
Print '</p>"'; 
} 
当然 ,仍然 可 以 使 用 foreach 循环 访问 多 维 数组 ,如 果 需 要 的 话 ,可 把 一 个 foreach 循 
环 嵌 套 在 另 一 个 内 部 。 
(8) 结束 PHP 代码 片段 并 且 完成 HTML 页 面 ,将 文件 保存 为 interests. php ,并 保存 
在 启用 了 PHP 服务 器 上 适当 的 目录 中 ,然后 在 Web 浏览 器 中 进行 测试 (参见 图 4-4) 。 


我 喜欢 的 一 首 歌 是 从 从 必 。 

我 喜欢 的 电影 是 将 下 会 矶 
‖| 我 喜欢 的 书 是 品 三 向。 

我 喜欢 的 第 1 个 是 小 小 虫 


我 喜欢 的 第 2 个 是 东风 破 
我 喜欢 的 第 3 个 是 waka waka 


1 个 是 钢 
我 喜欢 的 第 2 个 是 3] 
我 喜欢 的 第 3 个 是 阿 


book 

我 喜欢 的 第 1 个 是 哈 利 波 特 
我 喜欢 的 第 2 个 是 暮 光 之 城 
我 喜欢 的 第 3 个 是 品 三 国 


图 4-4 foreach 的 嵌 套 实现 对 二 维 数组 中 每 个 元 素 的 访问 


可 以 使 用 一 系列 嵌 套 的 array() 调 用 的 方式 在 一 个 语句 中 创建 多 维 数组 (用 以 代替 本 
示例 中 使 用 多 个 步骤 的 方式 ) 。 但 是 .并 不 推荐 这 样 做 ,因为 随 着 语句 中 肉 套 数量 的 增加 ， 
引发 语法 错误 的 可 能 性 就 越 大 。 
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这 关 


尽管 本 示例 中 的 所 有 子 数 组 都 有 相同 的 结构 (用 数值 作为 索引 ,并 且 有 3 个 元 素 ) ,但 


不 是 多 维 数组 所 必需 的 。 
4.3 数组 应 用 


4.3.1 基本 数组 函数 


1. 获取 数组 个 数 
要 确定 数组 中 的 元 素 个 数 ,可 以 使 用 count() 或 sizeofO 〇 ) 函 数 ,这 两 个 函数 是 同 义 的 。 
$num= count ($array); 


count() 是 经 常用 到 的 一 个 函数 ,其 功能 是 返回 一 个 数组 的 长 度 , 它 会 遍历 整个 数组 


然后 求 出 元 素 个 数 。 例 如 : 


$A=array ("one", "two", "three", "four"); 
$num= count ($A); 
for($i=0;$i<$num$i++){ 

echo "$A[$i]"."<br /> "7 
} 


上 述 代码 与 foreach 循环 遍历 数组 有 异曲同工 之 处 ,其 功能 是 遍历 整个 数组 并 输出 


每 个 数组 元 素 。count() 函 数 经 常 运 用 于 for 循环 中 。 


2. 在 数组 中 搜索 给 定 的 值 
检查 数组 里 面 是 否 包含 给 定 的 值 , 可 以 使 用 in_array() 来 轻松 地 完成 这 项 功能 。 


in_array($valuev 5 array,true); 


如 果 给 定 的 值 $ value 存在 于 数组 $ array 中 , 则 返回 true。 如 果 第 三 个 参数 设置 为 


true, 那 么 函数 只 有 在 元 素 存 在 于 数组 中 且 数 据 类 型 与 给 定 值 相同 时 才 返 回 true。 如 果 
没有 在 数组 中 找到 参数 , 则 函数 返回 false。 如 果 $ value 参数 是 字符 串 , 且 第 三 个 参数 设 
置 为 true, 则 搜索 区 分 大 小 写 。 例 如 : 


$ os=array ("Mac", "NT", "Windows", "Linux"); 
if (in array ("Linux", $03)){ 
echo "找到 Linux"; 
Jelse{ 
echo " 没 找到 Linux"; 
} 


上 述 代码 的 功能 是 在 $ os 数组 中 搜索 “Linux” 字 符 串 。 
3. 创建 指定 范围 的 元 素 的 数组 
range() 可 用 来 快速 创建 连续 字母 或 数字 的 数组 。 


第 4 章 数 组 


range (first, second, ) 


该 函数 创建 一 个 数组 ,包含 从 first 到 second (包含 first 和 second) 之 间 的 整数 或 字 
符 。 如 果 second 比 first 小 , 则 返回 反 序 的 数组 。 例 如 : 


$ number=range (0, 5); 

上 述 代码 将 实现 快速 创建 一 个 数字 数组 ,长 度 为 6 ,元 素 的 值 为 0 一 5。 

range() 函 数 还 可 用 于 脚本 3-2 中 使 用 循环 生成 下 拉 菜 单 , 可 用 该 函数 快速 创建 年 月 
日 的 连续 数字 数组 ,通过 foreach 进行 遍历 。 


4.3.2 数组 与 字符 串 的 转换 


字符 串 和 数组 都 非常 常用 ,在 这 两 种 形式 之 间 进 行 转换 有 两 个 函数 : 第 一 个 是 
implode() , 它 用 来 将 数组 转换 为 字符 串 ;第 二 个 是 explode() 用 来 做 相反 的 处 理 。 
使 用 这 些 函 数 的 原因 是 : 
。 将 数组 转换 为 字符 串 是 为 了 能 够 将 值 附 加 在 URL 上 进行 传递 ,因为 不 能 用 数组 
实现 。 
。 将 数组 转换 为 字符 串 是 为 了 将 信息 更 方便 地 储存 在 数据 库 中 。 
。 将 字符 串 转换 为 数组 是 为 了 将 一 个 用 逗号 分 隔 的 文本 区 ,例如 表单 中 的 关键 字 搜 
索 区 域 , 转 化 为 相互 独立 的 形式 。 
使 用 explode() 的 语法 如 下 : 
$array= explode ($ separator，5$ string)? 
使 用 implode() 的 语法 如 下 : 
$ string=implode ($ glue, $array); 
使 用 和 理解 这 两 个 函数 的 关键 之 处 是 分 隔 符 (separator) 和 胶合 (glue) 关 系 。 当 把 一 
个 数组 转变 成 一 个 字符 串 时 ,将 会 设置 胶合 (glue) 一 一 将 被 插入 到 生成 字符 串 中 的 数组 
值 之 间 的 字符 或 代码 。 相 反 , 当 把 字符 串 转变 成 数组 时 ,要 指定 分 隔 符 (separator) , 它 是 
描绘 生成 数组 中 的 不 同 元 素 之 间 的 连接 的 代码 。 
separator 分 隔 符 指明 了 一 个 或 多 个 字符 ,用 来 区 分 一 个 值 的 结束 和 另外 一 个 值 开 
始 。 通 常情 况 下 分 隔 符 是 一 个 逗号 ,一 个 制 表 符 或 者 一 个 空格 。 因 此 explode 可 能 如 此 
使 用 : 


$array=explode(',', $string); 


或 

Sarray-explode(' ', $string); 

为 了 将 一 个 数组 转换 为 字符 串 ,需要 定义 用 什么 作为 分 隔 符 , 也 就 是 之 间 的 连接 符 ， 
PHP 将 处 理 剩 下 的 工作 。 因 此 implode 可 能 如 此 使 用 : 


$ string=implode(',', $array); 


qi! 
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或 

$ string=implode(' ', $array); 

例如 : 

$ stringl= 'Mon- Tue- Wed- Thur- Fri'7 

S$days_array- explode('- ', $stringl); 

$ days_array 变量 现在 是 一 个 有 5 个 元 素 的 数组 ,其 元 素 Mon 的 索引 为 0,Tue 的 索 
引 为 1 ,等 等 。 

5$ string2= implode(', ', $days array); 

$ string2 变量 现在 是 一 个 逗号 分 隔 的 一 个 星期 中 各 天 的 列表 : Mon、Tue、 Wed、 
Thur 和 Fri。 


4.4 项 目 训练 


简易 判断 文件 格式 


4.4.1 项 目 说 明 


在 表单 上 传 等 操作 过 程 中 ,浏览 器 上 传 的 文件 安全 性 检查 非常 重要 , 若 上 传 了 恶意 的 
非法 文件 ,可 能 对 服务 器 造成 不 可 预期 的 影响 。 因 此 ,检查 上 传 文件 的 类 型 是 防止 恶意 软 
件 的 一 个 有 效 步骤 。 在 PHP 中 ,判断 文件 类 型 有 多 种 方法 ,这 里 要 求 提取 文件 的 后 组 名 
(扩展 名 ) 并 判断 文件 类 型 是 否 符合 要 求 。 


4.4.2 设计 思路 


一 般 情况 下 ,服务 器 允许 上 传 的 文件 类 型 不 止 一 种 ,因此 要 事先 制作 一 个 允许 列表 ， 
包含 多 种 可 上 传 的 文件 扩展 名 。 获 取 文件 名 后 ,使 用 explode 函数 分 割 文件 名 ,取得 扩展 
名 ,然后 与 允许 列表 比 对 。 例 如 ,文件 名 为 myfile. doc, 人 允许 的 文件 后 级 名 列表 为 jpg/gif/ 
png, 获 得 后 级 名 “doc” 后 ,在 允许 列表 中 搜索 是 否 包 含 该 扩展 名 。 


4.4.3 设计 过 程 


根据 文件 扩展 名 判断 文件 类 型 ,需要 使 用 之 前 的 explode() ,count() ,inarray() 等 数 
组 应 用 函数 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 4-5)。 


脚本 4-5 获取 文件 扩展 名 判断 是 否 允 许 上 传 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN”"http://www-w3.org/TR/ 
xhtml1/DTD/xhtml1- transitional.dtd"> 
2 <html xmlns= "http://www-w3.org/1999/xhtml"> 
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3 <head> 

4 <meta http-equiv="Content- Type”" content="text/html; charset=gb2312" /> 
5 <title> 测 试 文件 格式 < /title> 

6 </head> 
8 

9 


<body> 

<?php 
10 $filename= "myfile.doc"7 
11 $filetypes=array('jpg', 'png', 'gif'); 
12 echo "文件 名 为 :$ filename<br /> \n"; 
14 
15 $file=explode (".",$ filename); 
16 Print r($file); 
至 echo "<br /> "7 
18 
19 $newtype=$ file[count ($ file)-1]; 
20 if(!in array($ newtype, $ filetypes)) 
C2 echo "文件 格式 错误 ,必须 是 图 片 !"; 
22 else 
23 echo "文件 格式 符合 要 求 , 可 以 上 传 !"; 
24 ?> 
25 </body> 
26 </htm> 


(2) 设置 初始 文件 名 称 , 以 及 允许 上 传 的 文件 格式 ,多 种 文件 扩展 名 可 定义 为 数组 
格式 。 

$ filename= "myfile.doc"; 

$ filetypes=array('jpg', 'png', 'gif'); 

(3) 使 用 explode 函数 ,以 *. ”为 分 隔 符 ,将 字符 串 的 文件 名 转换 为 一 个 数组 。 


$ file=explode (".",$ filename); 


(4) 使 用 count() 函 数 ,获取 分 割 后 的 数组 的 最 后 一 个 元 素 的 键 ,以 此 提取 出 文件 扩 
展 名 。 


Snewtype=$ file[count ($file)-1]; 
(5) 判断 取得 的 文件 扩展 名 是 否 在 允许 列表 中 ,这 里 使 用 了 in_array() 函 数 。 


if(!in array ($newtype, $filetypes)) 
echo "文件 格式 错误 ,必须 是 图 片 !"; 
else 


echo "文件 格式 符合 要 求 , 可 以 上 传 !"; 


(6) 结束 PHP 代码 片段 并 且 完 成 HTML 页 面 ,将 文件 保存 为 checkfile. php .并 保存 
在 启用 了 PHP 服务 器 上 适当 的 目录 中 .然后 在 Web 浏览 器 中 进行 测试 (参见 图 4-5)。 
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i 


文件 名 为 ，myfile. doc 
Array ( [0] => myfile [1] => doc ) 
文件 格式 错误 ， 必 须 是 图 片 ! 


4-5 使 用 多 个 数组 应 用 函数 实现 文件 扩展 名 判断 


本 章 小 结 


本 章 介绍 了 PHP 中 数组 的 概念 ,详细 介绍 了 数组 的 类 型 ,如 何 创 建 数组 .访问 数组 
元 素 、 打 印 数组 及 遍历 数组 元 素 , 并 介绍 了 多 维 数组 的 应 用 及 PHP 中 数组 常用 的 各 种 内 


置 函数 。 
重点 回顾 
1. 数组 的 类 型 。 
2. 如 何 遍 历数 组 。 
3. 数组 应 用 函数 。 
本 章 实 训 
【 实 训 1】 


新 建 score. php, 统 计 你 的 实 训 成 绩 , 使 用 数组 保存 实验 和 成 绩 , 先 输出 数组 内 容 和 结 
构 , 然 后 使 用 表格 输出 。 网 页 运行 效果 如 图 4-6 所 示 。 

【 实 训 2】 

新 建 movie. php, 定 义 一 个 包含 5 个 元 素 的 多 维 数组 ,内 容 为 前 5 名 电影 排名 ,每 部 
电影 包含 名 称 和 imdb 评分 , 先 使 用 print_r 输出 ,再 输出 为 表格 格式 。 网 页 运行 效果 如 
图 4-7 所 示 。 


Array 〔 [ 实 训 一 ] => 10 [ 实 训 二 ] => 8.5 [ 实 训 三 ] => 
9 [ 实 训 四 ] => 10 ) 


图 4-6 ”score. php 的 运行 效果 


Array ( [0] => Array ( [name] => 地 心 引力 [imdb] => | 
8.3 ) [1] =》Array ( [name] =》 雷神 2， 黑暗 世界 

[indb] =》7.8 ) [2] =》Array ( [name] =》 金 婵 脱 元 
[imdb] => 7.8 ) [3] => Array( [name] => 赤 炮 战场 2 
[imdb] => 7.1 ) [4] =》Array ( [name] =》 精灵 旅社 
[imdb] => 7.8 ) ) 


| 电影 名 称 [三 imdb 评分 


有 邢 必 5 万 8.3 
备 神 2， 黑暗 世界 


[ 潮 炮 战场 2 
| 精灵 旅社 


图 4-7 movie. php 的 运行 效果 


第 4 章 数 组 


第 5 章 也 数 


函数 可 以 简单 地 分 为 两 大 类 : 一 类 是 系统 内 置 函数 ; 另 一 类 是 用 户 自 定义 函数 。 对 
于 系统 函数 ,可 以 在 需要 时 直接 选择 使 用 ;而 对 于 用 户 自 定义 函数 ,首先 要 定义 ,然后 才能 
使 用 。 


5.1 创建 和 调用 自 定义 函数 


5.1.1 自 定义 函数 


PHP 内 置 函 数 允许 和 文件 进行 交互 、 使 用 数据 库 、 创 建 图 形 , 还 可 以 连接 其 他 的 服务 
器 。 但 是 ,在 实际 工作 中 ,有 许多 时 候 所 需要 的 东西 是 语言 的 创建 者 无 法 预见 到 的 。 任 何 
好 的 编程 语言 都 有 一 个 重要 特点 ,就 是 能 够 声明 并 编写 函数 ,因此 可 以 编写 自己 的 函数 来 
完成 任何 所 需 的 任务 。 基 本 上 ,函数 就 是 可 以 调用 以 得 到 所 需 结果 的 代码 块 ,这 些 代码 块 
可 能 是 已 有 函数 和 自己 逻辑 的 混合 体 , 通 过 它 来 完成 任务 。 

如 果 你 正在 为 一 个 任务 编写 一 段 代码 ,而 很 可 能 这 段 代码 将 在 一 个 脚本 的 多 处 或 是 
多 个 脚本 中 都 要 使 用 ,那么 最 明智 的 方法 是 将 这 段 代 码 声 明 为 函数 。 

声明 一 个 函数 可 以 让 我 们 像 内 置 函 数 那样 使 用 自己 的 代码 ,PHP 中 可 以 向 函数 传递 
值 ,也 可 以 从 中 返回 值 。 只 要 简单 地 调用 这 个 函数 并 提供 给 它 必需 的 参数 。 这 就 意味 着 ， 
在 整个 脚本 中 ,都 可 以 调用 和 多 次 重复 使 用 相同 的 函数 ,能 够 清理 大 量 宛 余 的 代码 。 

1. 自 定义 函数 

建立 自 定义 函数 的 语法 如 下 : 

function function name([$arg 1][, $arg 2][, ..][, $arg n]){ 


statement 


} 


定义 函数 是 需要 使 用 function 关键 字 , 之 后 是 函数 名 。 

函数 名 与 变量 一 样 , 也 遵循 同样 的 命名 约定 ,不 过 与 变量 不 同 的 是 ,函数 名 的 最 前 面 
不 能 加 $ 。 函 数 名 中 的 第 一 个 字母 同样 不 能 是 数字 ,而 可 以 是 任何 字母 或 下 划 线 。 后 面 
的 字符 则 可 以 是 字母 .数字 和 下 划 线 的 任意 组 合 。 不 可 以 为 自 定义 函数 使 用 现 有 的 函数 
名 (print\echo \isset 等 ) 。 


$arg_1 到 $arg_n 为 函数 的 可 选 参 数列 表 . 不 同 的 参数 之 间 用 逗号 分 隔 。 在 函数 内 
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部 可 以 放置 任何 有 效 的 PHP 代码 ,从 生成 HTML 代码 到 预先 格式 化 计算 ,甚至 包括 其 
他 函数 和 类 定义 。 例 如 : 


<?Pphp 

function Sum($a,$b){ 

$c=$at$b; 

return $c; 

} 

echo Sum(10,100); // 输 出 :110 
入 


注意 ,在 PHP 3 中 ,函数 必须 在 被 调用 之 前 定义 。 而 在 PHP 4 之 后 则 不 再 有 这 样 的 
限制 。 上 面 的 一 段 代码 也 可 以 写成 : 

<?php 

echo Sum(10,100); // 输 出 :110 

function Sum($a,$b){ 

$c=$at$b; 

return $c; 


} 

2 

2. 理解 字母 大 小 写 和 函数 名 称 

请 注意 ,函数 调用 将 不 区 分 字母 大 小 写 .所 以 调用 function_name()、Function_Name() 或 
FUNCTION_NAME() 都 是 有 效 的 ,而 且 都 将 返回 相同 的 结果 。 可 以 按照 便于 自己 阅读 
的 方式 任意 使 用 大 小 写 , 但 应 该 尽量 保持 一 致 。 本 书 和 大 多 数 PHP 文档 使 用 的 命名 惯 
例 是 : 所 有 字母 都 用 小 写 。 

应 注意 ,函数 名 称 和 变量 名 称 是 不 同 的 ,这 一 点 很 重要 。 变 量 名 是 区 分 大 小 写 的 ,所 
以 $ Name 和 $name 是 两 个 不 同 的 变量 ,但 Name() 和 name() 则 是 同一 个 函数 。 

3. 创建 自己 的 函数 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-1) 。 


脚本 5-1 这 个 用 户 定义 的 函数 用 于 创建 一 系列 下 拉 菜单 。 


1 < !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional.dtd"> 

3 <html xmlns= "http://www-w3-org/1999/xhtml"> 

4 <head> 

5 <meta http- equiv= "Content- Type" content="text/html; charset=gb2312" /> 
6 ”<title> 数 组 排序 < /title> 

7 </head> 

8 <body> 

9 


10 <?php #Script 5-1- dateform.php 


7 
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11 // 定 义 函数 ,生成 年 月 日 三 个 下 拉 菜 单 
12 function make_calendar pulldowns (){ 


13 echo '< select name= "Year"> "7 

14 $ year= 2005; 

15 while($ year <=2015){ 

16 echo "< option value= \"$ year\"> $ year< /option> \n"; 
17 $yeart+?; 

18 } 

19 echo '< /select> 年 '; 

20 

21 echo '< select name= "month"> "7 

Se Smonth= 17 

23 dof 

24 echo "< option value= \"$month\"> $month< /option> \n"; 
25 $montht+? 

26 J}while($ month <=12); 

27 echo '</select> 月 '; 

28 

2 echo '< select name= "day"> '; 

30 for($day=1; $day <=31; $day++){ 
31 echo "< option value=\"$ day\"> $ day< /option> \n"; 
E74 } 

33 echo '< /select> 日 '; 

34 ]}// 函 数 定义 结束 

35 ?> 

36 请 选择 一 个 日 期 : 

37 <formaction="" method=- "post"> 

38 <?php 

39 // 调 用 函数 

40 make_calendar pulldowns ()? 

41 ?> 

42 </form> 

43 </body> 

44 <“/html> 


(2) 开始 定义 一 个 新 函数 。 
function make_calendar pulldowns (){ 


这 里 编写 的 函数 将 会 生成 用 于 选择 月 份 . 天 和 年 份 所 需 的 表单 下 拉 菜 单 , 就 像 
calendar. php 中 那样 (参见 脚本 3-2)。 这 个 函数 的 名 称 显然 指明 了 其 用 途 。 

尽管 不 需要 这 样 做 ,但 是 通常 把 函数 定义 放 在 靠近 脚本 顶部 的 位 置 , 或 者 放 在 一 个 单 
独 的 文件 中 。 

(3) 生成 下 拉 莱 单 ,此 处 代码 参见 脚本 3-2。 

(4) 关闭 函数 定义 ,并 退出 PHP 代码 块 。 


第 5 章 函 数 A) 


]}// 函 数 定义 结束 

了 3 

在 函数 定义 的 末尾 放置 一 条 注释 ,这样 就 可 以 知道 一 个 定义 开始 和 终止 于 何 处 。 
(5) 创建 表单 并 调用 函数 。 

请 选择 一 个 日 期 : 


< form action="" method= "post"> 

<?php 

// 调 用 函数 

make_calendar pulldowns (); 

?> 

</form> 

这 段 代码 将 会 创建 一 个 表单 的 标签 ,并 进入 新 的 PHP 代码 块 ,调用 make_calendar_ 
pulldowns() 函 数 的 最 终结 果 是 创建 3 个 下 拉 菜 单 。 

(6) 完成 脚本 ,将 文件 另存 为 dateform. php, 存 放 在 Web 目录 中 ,并 在 Web 浏览 器 
中 测试 它 ( 参 见 图 5-1)。 


5-1 这 些 下 拉 菜 单 由 用 户 自 定义 的 函数 完成 


如 果 出 现 call to undefined function function_name( 调 用 了 未 定义 的 函数 function_ 
name) 错 误 , 则 意味 着 正在 调用 一 个 未 定义 的 函数 。 如 果 拼 错 了 函数 的 名 称 (在 定义 或 调 
用 它 时 ) 或 者 无 法 包含 定义 函数 的 文件 ,就 可 能 发 生 这 种 错误 。 

由 于 用 户 定义 的 函数 会 占用 一 些 内 存 , 所 以 在 使 用 这 样 一 个 函数 时 应 该 小 心间 慎 。 
一 般 的 规则 是 ,最 好 把 这 些 函 数 用 于 可 能 在 脚本 或 Web 站 点 的 多 个 位 置 执行 的 大 段 
代码 。 


5.1.2 创建 带 参数 的 函数 


就 像 PHP 的 内 置 函 数 一 样 ,可 以 编写 带 参数 (argument. 也 称 为 parameter) 的 函数 。 
例如 ,isset() 函数 接受 要 测试 的 变量 名 称 作 为 参数 。count() 接 受 一 个 要 确定 其 长 度 的 数 
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组 作为 参数 。 

通过 参数 列表 可 以 传递 信息 到 函数 。PHP 支持 按 值 传递 参数 ,通过 引用 传递 以 及 默 
认 参 数 。 在 PHP 4 和 后 续 版 本 中 还 增加 了 对 可 变 长 度 参数 列表 的 支持 ,在 PHP 3 中 可 
以 通过 传递 一 个 数组 参数 达到 类 似 的 效果 。 

一 个 函数 可 以 带 有 任意 数量 的 参数 ,但 是 列 出 它们 的 顺序 很 重要 。 为 了 应 用 这 些 参 
数 ,可 把 变量 添加 到 自 定义 的 函数 中 : 


function print hello($ first, $1ast){ 
//Function code. 
} 


一 旦 定义 了 函数 ,就 可 以 像 调 用 PHP 中 的 其 他 任何 函数 那样 调用 这 个 函数 ,并 把 值 
或 变量 发 送 给 它 : 

print_hello(' 李 ',' 宇 春 '); 

$1astname= ' 宇 春 '; 

print_hello(' 李 ', $1astname); 


定义 带 参 数 的 函数 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-2) 。 


脚本 5-2 sum. php 脚本 现在 使 用 一 个 函数 来 执行 它 的 计算 。 与 用 户 定义 的 函数 
make_calendar _ pulldowns() 不 同 的 是 ,这 个 函数 带 有 参数 。 


< !DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN” 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 


1 
3 
4 
5 <meta http-equiv= "Content- TYPpe"”content= "text/html1; charset=gb2312" /> 
6 <title> 求 和 </title> 

7 </head> 

8 <body> 

3 

10 <?php #Script 5- 2- sum.php 

11 // 定 义 函数 , 求 两 数 之 和 

12 function sum($a, $b){ 


3 $ total= $at+$b; 

14 echo "x+y=$ total"; 
15 } 

16 $x=100; 

和 二 $1; 

18 Sum($x, Sy); 

19 ?> 


20 </body> 
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21 </htm> 
(2) 这 个 函数 将 实现 两 个 变量 求 和 的 计算 ,然后 打印 出 结果 , 它 带 有 两 个 参数 。 
function sum($a, Sb){ 

$total=$at$b; 

echo "x+y=$ total"; 
} 
(3) 设 定 变量 初 值 ,然后 调用 自 定义 函数 。 


$x=100; 

$y=1; 

sum($x, $y); 

当 调 用 也 数 时 ,将 会 给 它 传递 两 个 参数 ,其 中 每 个 参数 都 是 一 个 变量 。 将 把 $ x 的 值 
赋予 函数 的 $a 变量 ,把 $y 的 值 赋予 函数 的 $b 变量 。 

(4) 将 文件 另存 为 sum. php ,存放 在 Web 目录 中 ,并 在 Web 浏览 器 中 测试 它 ( 参 见 
图 5-2) 。 


5-2 ”使 用 用 户 定义 的 函数 执行 计算 


与 PHP 中 的 任何 函数 一 样 ,如 果 无 法 发 送 正确 数量 的 参数 ,就 会 导致 一 个 错误 ,如 
图 5-3 所 示 。 


Warning: Nissing argument 2 for sum(), called in 

| C:\AppServ\wmm\book\chO5\sum. php on line 17 and 
defined in C:\AppServ\www\book\ch05\sum. php on 
line 11 


图 5-3 无 法 给 函数 发 送 正确 数量 和 类 型 的 参数 将 会 引发 错误 


s.1.3 设置 默认 的 参数 值 


定义 自己 的 函数 的 另 一 个 变 体 是 预先 设置 参数 的 值 。 要 这 样 做 ,可 以 在 函数 定义 时 ， 
给 参数 赋 初 值 : 


function greet ($name, $msg= "Hello'){ 
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echo "Smsg $name!"; 


} 


设置 默认 参数 值 的 最 终结 果 是 , 当 调用 函数 时 ,特定 参数 变 为 可 选 的 。 如 果 把 一 个 值 
传递 给 它 , 就 会 使 用 传递 的 值 ;否则 ,就 会 使 用 默认 值 。 

可 以 根据 需要 为 多 个 参数 设置 默认 值 ,只 要 这 些 参 数 出 现在 函数 定义 的 最 后 面 即 可 。 
换 句 话说 ,必需 的 参数 应 该 总 是 出 现在 最 前 面 。 

利用 刚才 定义 的 示例 函数 ,下 面 的 任何 一 个 函数 都 将 会 工作 : 


greet ($ surname, $message); 

greet (" 张 明 )7 

greet (' 李 丽 '，'Good evening'); 

不 过 ,只 有 greet() 不 会 工作 ,并 且 如 果 不 给 $ name 传递 一 个 值 ,就 无 法 给 $ msg 传 
递 一 个 值 ,因为 必须 按 顺 序 传递 参数 值 ,不 能 跳 过 必需 的 参数 。 例 如 : 


<?php 

function makecoffee ($ type= "cappuccino") { 
echo "给 我 来 一 杯 $ type。<br />"; 

} 

makecoffee (); 

makecoffee ("espresso"); 


2 
这 段 代 码 的 输出 是 : 


给 我 来 一 杯 cappuccino。 
给 我 来 一 杯 espresso。 


在 这 段 代码 中 ,函数 的 参数 $type 有 默认 值 cappuccino ,在 函数 调用 时 , 若 不 提供 实 
际 的 参数 ,即使 用 makecoffee() , 则 参数 $type 使 用 默认 值 cappuccino, 如 提供 实际 参数 ， 
即 makecoffee("espresso") , 则 参数 $ type 的 值 是 espresso。 

为 了 不 给 函数 的 参数 传递 任何 值 ,可 以 使 用 空 字符 串 ()、NULL 或 FALSE。 


5.1.4 ”从 函数 返回 值 


讨论 的 用 户 定义 的 函数 的 最 后 一 个 属性 是 返回 值 。 函 数 可 以 有 返回 值 , 也 可 以 没有 
返回 值 。 例 如 ,print() 将 会 返回 1 或 0 来 指示 它 是 否 成 功 执行 ,而 echo() 则 不 会 返回 值 。 
另 一 个 例子 是 ,count() 函数 会 返回 一 个 与 数组 中 的 元 素 个 数 相关 的 数字 。 

要 让 函数 返回 一 个 值 ,可 以 使 用 return 语句 。 


function find sign($month, $day){ 
//Function code. 
return $ sign; 


} 
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这 个 函数 可 以 返回 一 个 值 (比如 一 个 字符 串 或 一 个 数字 ) ,或 者 一 个 变量 (这 个 函数 已 
经 产生 了 它 的 一 个 值 )。 当 调用 这 个 返回 值 的 函数 时 ,可 以 将 函数 结果 赋 子 一 个 变量 。 


$my_sign= find sign('October', 23); 


或 者 把 它 用 作 调 用 另 一 个 函数 时 的 参数 。 


print find sign('October', 23); 

任何 类 型 都 可 以 返回 ,其 中 包括 数组 和 对 象 。 这 导致 函数 立即 结束 它 的 运行 ,并 且 将 
控制 权 传递 回 它 被 调用 的 行 。 

让 函数 返回 一 个 值 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-3) 。 


脚本 5-3 sum() 函 数 将 执行 计算 ,并 返回 计算 的 结果 。 


22 


< !DOCTYPE html PUBLIC "~ //W3C//DITD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.0rg/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312"” /> 
<title> 求 和 < /title> 
</head> 
<body> 


< ?php #Script 5- 3- returnsum.php 
// 定 义 函 数 , 求 两 数 之 和 
function sum($a, $b){ 
$ total=$at+$b; 
return $ total; 
} 
$x=100; 
$y-1; 
$totalvalue= sum($ x, $y); 
echo "x+y=$ totalvalue"; 
?> 
< /body> 
</htm> 


(2) 修改 sum 了 〇 函数 ,将 echo 语句 改 为 return 语句 ,将 总 和 变量 的 值 返回 。 


function sum($a, $b){ 


$total=$at$b; 
return $ total; 
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(3) 由 于 函数 现在 会 返回 值 , 而 不 会 打印 计算 结果 ,所 以 需要 把 函数 调用 赋予 一 个 变 
量 ,使 得 以 后 可 以 在 脚本 中 打印 总 额 。 


$ totalvalue= sum($ x, $y);? 
(4) 添加 一 条 新 的 echo() 语 句 , 用 于 打印 结果 。 
echo "x+ y=$ totalvalue"; 


因为 这 个 函数 只 会 返回 一 个 值 ,所 以 必须 把 新 的 echo() 语 句 加 入 主 代码 中 。 

(5) 保存 文件 ,存放 在 Web 目录 中 ,并 在 Web 浏览 器 中 测试 它 , 如 图 5-4 所 示 , 用 户 
定义 的 函数 现在 将 返回 (而 不 是 打印 ) 结 果 , 但 是 这 种 改变 对 用 户 所 看 到 的 内 容 没有 任何 
影响 。 


5-4 函数 返回 值 


需要 注意 的 是 ,return 语句 会 终止 代码 执行 ,因此 ,在 执行 return 语句 之 后 ,永远 也 
会 运行 函数 内 的 任何 代码 。 

另外 ,一 个 函数 可 以 具有 多 条 return 语句 ,例如 ,在 让 条 件 语句 或 switch 语句 中 ,但 
是 ,至 多 只 会 调用 其 中 的 一 条 return 语句 。 例 如 ,函数 通常 是 以 下 结构 : 


function some function(){ 
if(/* condition * /){ 
return TRUE; 
} else { 
return FALSE; 
} 
} 


要 让 一 个 函数 返回 多 个 值 ,可 以 使 用 array() 函 数 返 回 一 个 数组 。 这 是 一 个 小 技巧 。 


5.2 PHP 内 置 饮 数 


PHP 具有 许多 内 置 函数 ,可 以 应 对 几乎 所 有 的 需要 。PHP 之 所 以 简洁 易 用 ,也 是 因 
为 它 内 置 了 大 量 功 能 强大 的 函数 ,只 要 能 熟练 地 操作 这 些 函 数 , 就 可 以 通过 少量 的 代码 实 
现 复 杂 的 功能 。 

PHP 提供 了 很 多 内 置 函 数 .这 些 函 数 可 以 在 程序 中 直接 使 用 .按照 其 实现 的 功能 划 
分 为 很 多 个 函数 库 。 另 外 ,还 可 以 通过 扩展 模块 的 方式 引入 更 多 的 其 他 函数 。 内 置 函 数 
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库 非 常 庞 大 , 酚 数 非常 多 , 足 有 上 千 种 ,在 后 面 的 章节 中 会 陆续 介绍 常用 的 函数 。 熟 练 掌 
握 自 定义 函数 和 内 置 函 数 ,可 以 极 大 地 提高 编程 人 员 的 开发 效率 ,迅速 提高 编程 水 平 。 

内 置 函数 涉及 PHP 应 用 的 各 个 方面 ,数量 众多 ,本 节 仅 仅 介绍 几 个 常见 应 用 中 的 常 
用 函数 ,更 多 的 函数 说 明 请 参考 PHP 手册 。 


5, 


| 


2.1 常见 的 基本 函数 
生成 随机 数 


rand() 也 数 可 以 返回 随机 整数 。 语 法 如 下 : 


rand (min,max); 


min 和 max 为 两 个 可 选 参数 ,用 于 规定 随机 数 产生 的 范围 。 如 果 没 有 提供 可 选 参数 
min 和 max,rand() 返 回 0 到 RAND_MAX 之 间 的 伪 随 机 整数 。 例 如 , 想 要 5 一 15( 包 括 5 
和 15) 之 间 的 随机 数 ,用 rand(5, 15) 。 

在 某 些 平台 下 (例如 , Windows)RAND_MAX 只 有 32 768。 如 果 需 要 的 范围 大 于 


32 768， 


那么 指定 min 和 max 参数 就 可 以 生成 大 于 RAND_MAX 的 数 。 


Rand() 函数 只 可 生成 随机 整数 ,在 网 页 验证 身份 等 范例 中 ,常见 的 验证 码 功能 要 求 
能 够 生成 随机 字符 串 ,包含 大 小 写字 母 和 数字 。 

脚本 5-4 就 利用 rand() 函 数 来 实现 随机 字符 串 。 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-4) 。 


脚本 5-4 利用 rand() 函 数 生成 随机 字符 串 。 


1 
2 
3 
4 
5 
6 
7 
8 
9 
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< !DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title> 取 得 定常 的 随机 字符 串 < /title> 
</head> 
<body> 


< ?php #Script 5- 4- rand.php 

// 取 得 定常 的 随机 字符 串 ,包含 大 小 写字 母 和 数字 

$length=5; 

$string="'; 

$arr=array(a,b,c,d,e,f,g,h,i,j,k,l,mn,o,pd r,s,t,U VW, XYZ/AB,C,D,E,F,G,H,I, 
J,K,L,M,N,O,P,Q,R,S,T,U,V,W,X,Y,2,1,2,3,4,5,6,7,8,9,0); 


$arrlength= count ($arr); 
for($i=0;$ i<$ length;$ i++) 
$string .=$arr[rand (0,$arrlength- 1)]; 


86 PHP+MySQL 项 目 实例 开发 


19 echo $ string; 


20 ?> 

21 </body> 

22 </htm> 

(2) 初始 化 $length 变量 .定义 要 生成 的 随机 字符 串 的 长 度 ,这 里 定义 的 长 度 为 5。 
$ length=5; 


(3) 初始 化 $ string 变量 ,用 来 存放 要 生成 的 随机 字符 串 ,开始 时 为 空 字符 串 。 
$string=""; 
(4) 初始 化 $arr 数组 ,用 来 存放 所 有 的 大 写字 母 .小 写字 母 和 数字 。 


$arr=array(a,b,c,d,e,f,g,h,i,j,k,l,m,n,o,prqd, rr 3,t, UV,W, X,Y 2AB,C,D,E,F,G,H, TJ, K, 
LM,N,O,P,Q,R,S,T,U,V,W, X,Y,2,1,2,3,4,5,6,7,8,9,0); 


(5) 使 用 $ arrlength 变量 存放 $ arr 数组 的 长 度 ,通过 count() 函 数 获取 数组 长 度 。 
$arrlength= count ($arr); 


(6) 使 用 for 循环 生成 随机 字符 串 ,每 次 循环 通过 rand() 函 数 生成 一 个 随机 整数 , 范 
围 从 0 到 $arr 数组 长 度 减 1, 此 随机 整数 作为 $arr 数组 的 下 标 ,由 此 生成 随机 的 数组 元 
素 。 循 环 5 次 ,每 次 生成 一 个 数组 元 素 , 并 通过 “. = "运算 符 连接 生成 $ string 字符 串 ,最 
后 打印 输出 。 

for($i=0;$ i<$ length;$i++) 

$string .=$arr[rand(0,$ arrlength)]; 

echo $ string; 

(7) 完成 脚本 ,将 文件 另存 为 rand. php ,存放 在 Web 目录 中 ,并 在 Web 浏览 器 中 测 
试 它 (参见 图 5-5)。 


5-5 应 用 rand 函数 生成 的 5 位 随机 字符 串 


上 例 的 随机 字符 串 是 应 用 rand() 函 数 并 结合 数组 ,循环 生成 的 ,此 例 将 用 在 后 例 的 
生成 随机 数 验 证 码 图 片 范例 中 。 


2. 最 大 值 和 最 小 值 
max() 函数 可 返回 最 大 值 。 语 法 如 下 : 


max (XrY) 7 
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参数 x 和 y 都 是 必需 的 ,可 以 是 一 个 数 。max() 会 返回 参数 x 和 y 中 数值 最 大 的 值 。 

如 果 仅 有 一 个 参数 上 且 为 数组 ,max() 返 回 该 数组 中 最 大 的 值 。 如 果 第 一 个 参数 是 整 
数 . 字 符 串 或 浮 点 数 , 则 至 少 需要 两 个 参数 而 max() 会 返回 这 些 值 中 最 大 的 一 个 。 可 以 比 
较 无 限 多 个 值 。 

使 用 max() 来 返回 两 个 指定 的 数 中 的 最 大 值 ,如 下 例 : 


<?php 
echo (max (5,7)); 

echo (max (7.25,7.30)); 
2 


输出 结果 为 ， 


这 


PHP 会 将 非 数 值 的 字符 串 当 成 0, 但 如 果 这 个 正 是 最 大 的 数值 , 则 仍然 会 返回 一 个 
字符 串 。 如 果 多 个 参数 都 求 值 为 0 且 是 最 大 值 , 则 max() 会 返回 其 中 数值 的 0, 如 果 参 数 
中 没有 数值 的 0, 则 返回 按 字母 表 顺 序 最 大 的 字符 串 。 

min() 函数 用 法 与 max( 〇 函数 相同 ,将 返回 参数 中 数值 最 小 的 值 。 

3. MDS 加 密 

MD5 算法 是 一 种 非常 优秀 的 加 密 算法 。MD5 的 全 称 是 Message-Digest Algorithm 
5, 在 20 世纪 90 年 代 初 由 MIT 的 计算 机 科学 实验 室 和 RSA Data Security Ine 发 明 ,经 
MD2 .MD3 和 MD4 发 展 而 来 。 

Message-Digest 泛 指 字 节 串 (Message) 的 Hash 变换 ,就 是 把 一 个 任意 长 度 的 字 节 串 
变换 成 一 定 长 的 大 整数 。 请 注意 这 种 变换 只 与 字 节 的 值 有 关 ,与 字符 集 或 编码 方式 无 关 。 

MD5 加 密 算法 的 特点 是 灵活 性 ,不 可 恢复 性 。MD5 将 任意 长 度 的 “ 字 节 串 " 变 换 成 
一 个 128 位 的 大 整数 ,并 且 它 是 一 个 不 可 逆 的 字符 串 变 换算 法 , 换 句 话说 就 是 ,即使 能 看 
到 源 程序 和 算法 描述 ,也 无 法 将 一 个 MD5 的 值 变换 回 原始 的 字符 串 ; 从 数学 原理 上 说 ， 
是 因为 原始 的 字符 串 有 无 穷 多 个 ,这 有 点 像 不 存在 反 函 数 的 数学 函数 。 

MD5 的 典型 应 用 是 对 一 段 Message( 字 节 串 ) 产 生 fingerprint (指纹 ), 以 防止 被 * 算 
改 ”。 例 如 ,我 们 将 一 段 话 写 在 一 个 叫 readme. txt 的 文件 中 ,并 对 这 个 readme. txt 产生 
一 个 MD5 的 值 并 记录 在 案 , 然 后 将 这 个 文件 传播 给 别人 ,别人 如 果 修改 了 文件 中 的 任何 
内 容 , 我 们 对 这 个 文件 重新 计算 MD5 时 就 会 发 现 原来 的 值 已 经 变化 。 如 果 再 有 一 个 第 
三 方 的 认证 机 构 , 用 MD5 还 可 以 防止 文件 作者 的 “抵赖 ”, 这 就 是 所 谓 的 数字 签名 应 用 。 

MD5 还 广泛 用 于 加 密 和 解密 技术 上 ,在 很 多 操作 系统 中 ,用 户 的 密码 是 以 MD5 值 
(或 类 似 的 其 他 算法 ) 的 方式 保存 的 ,用 户 登 录 的 时 候 , 系 统 是 把 用 户 输入 的 密码 计算 成 
MD5 值 ,然后 再 去 和 系统 中 保存 的 MD5 值 进行 比较 ,而 系统 并 不 “知道 ”用户 的 密码 是 
什么 。 

PHP 中 的 md5O 〇 函数 可 以 计算 字符 串 的 MD5 散 列 。md5() 函数 使 用 RSA 数据 安 
全 ,包括 MD5 报 文摘 译 算法 。 如 果 成 功 , 则 返回 所 计算 的 MD5 散 列 ;如 果 失 败 , 则 返 


回 
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False。 语 法 如 下 : 
md5 (string, charlist); 


参数 string 是 必需 的 ,是 规定 要 计算 的 字符 串 。 参 数 charlist 可 选 ,用 来 规定 十 六 进 
制 或 二 进 制 输出 格式 ,有 如 下 选择 : 


。 TRUE 一 一 原始 16 个 字符 二 进 制 格式 。 
。 FALSE 一 一 默认 32 个 字符 十 六 进 制 数 。 
例如 : 
<?Pphp 


$str="Hello"; 
echo md5 (5 str); 
?> 


输出 : 
8bla9953c4611296a827abf8c47804d7 


另外 ,在 当今 很 多 电子 商务 和 社区 应 用 中 ,管理 用 户 是 一 种 最 常用 的 基本 功能 ,尽管 
很 多 应 用 服务 器 都 提供 了 这 些 基 本 组 件 ,但 很 多 应 用 开发 者 为 了 管理 的 更 大 的 灵活 性 还 
是 喜欢 采用 关系 数据 库 来 管理 用 户 。 懒 惰 的 做 法 是 用 户 的 密码 往往 使 用 明文 或 经 简单 的 
变换 后 直接 保存 在 数据 库 中 ,因此 这 些 用 户 的 密码 对 软件 开发 者 或 系统 管理 员 来 说 毫 无 

假设 密码 明文 为 “Hello” ,数据 库 中 保存 了 “Hello" 的 密 文 , 那 么 每 次 用 户 输入 密码 
后 ,把 输入 的 密码 再 次 做 MD5 加 密 ,得 到 的 MD5 散 列 与 数据 库 中 已 保存 的 密 文 进行 比 
对 , 若 相 同 则 说 明 输 入 了 正确 的 密码 。 代 码 类 似 如 下 范例 : 

<?php 

$str="Hello"; 

echo md5 ($ str); 

if (md5 ($ str)== '8bla9953c4611296a827abf8c47804d7'){ 

echo "<br /> 您 的 输入 正确 !"; 
} 
?> 


输出 : 


8bla9953c4611296a827abf8c47804d7 
您 的 输入 正确 ! 


使 用 MD5 来 处 理 用 户 的 密码 ,使 得 管理 员 和 程序 设计 者 都 无 法 看 到 用 户 的 密码 , 尽 
管 他 们 可 以 初始 化 密码 。 但 重要 的 一 点 是 对 于 用 户 密码 的 保护 。 


5.2.2 日 斯 和 时 间 函 数 
在 开发 Web 程序 时 ,时 间 起 着 重要 的 作用 。 不 仅 在 数据 存储 和 显示 时 需要 日 期 和 时 
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间 的 管理 ,有 一 些 功能 模块 的 开发 ,时 间 通 常 是 至 关 重 要 的 。 例 如 ,网 页 静态 化 需要 判断 
缓存 时 间 、 计 算 页 面 访 问 消耗 的 时 间 、 在 不 同 的 时 间 段 提供 不 同 的 功能 等 。PHP 提供 了 
强大 的 日 期 和 时 间 处 理 功能 ,通过 时 间 和 日 期 函数 库 ,能 够 得 到 PHP 程序 在 运行 时 所 在 
服务 器 中 的 日 期 和 时 间 ,并 可 以 对 它们 进行 检查 和 格式 化 ,在 不 同 格式 之 间 进 行 转换 。 

1. UNIX 时 间 截 

UNIX 时 间 戳 是 保存 日 期 和 时 间 的 一 种 紧凑 简洁 的 方法 ,是 大 多 数 UNIX 系统 中 保 
存 当 前 日 期 和 时 间 的 一 种 方法 ,也 是 在 大 多 数 计算 机 语言 中 表示 日 期 和 时 间 的 一 种 标准 
格式 。 以 32 位 的 整数 表示 格林 尼 治 标准 时 间 , 例 如 ,使 用 整数 11230499325 表示 当前 时 
间 的 时 间 戳 。UNIX 时 间 戳 是 从 1970 年 1 月 1 日 零点 (UTC/GMT 的 午夜 ) 开 始 起 到 当 
前 时 间 所 经 过 的 秒 数 。1970 年 1 月 1 日 零点 作为 所 有 日 期 计算 的 基础 ,这 个 日 期 通常 称 
为 UNIX 纪元 。 

因为 UNIX 时 间 惟 是 一 个 32 位 的 数字 格式 ,所 以 特别 适用 于 计算 机 处 理 ,例如 计算 
两 个 时 间 点 之 间 相 差 的 天 数 。 另 外 ,由 于 文化 和 地 区 的 差异 ,存在 不 同 的 时 间 格 式 以 及 时 
区 的 问题 。 所 以 UNIX 时 间 戳 也 是 根据 一 个 时 区 进行 标准 化 和 设计 的 一 种 通用 格式 ,并 
且 这 种 格式 可 以 很 容易 地 被 转换 为 任何 格式 。 

也 因为 UNIX 时 间 惟 是 由 一 个 32 位 的 整数 表示 的 ,所 以 在 处 理 1902 年 以 前 或 
2038 年 以 后 的 时 间 时 ,将 会 遇 到 一 些 问题 。 另 外 ,在 Windows 下 ,由 于 时 间 戳 不 能 为 负 
数 ,如 果 使 用 PHP 中 提供 的 时 间 蕉 函 数 处 理 1970 年 之 前 的 日 期 ,就 会 发 生 错 误 。 要 使 
PHP 代码 具有 可 移植 性 ,必须 记 住 这 一 点 。 

在 PHP 中 ,如 果 需 要 将 日 期 和 时 间 转 变 成 UNIX 时 间 惟 ,可 以 调用 mktime() 函 数 。 
语法 如 下 : 

mktime ([int hour [, int minute [, int second [, int month [, int day [, int year ]]]]]]); 


该 函数 中 所 有 参数 都 是 可 选 的 ,如 果 参 数 为 空 , 则 默认 将 当前 时 间 转 变 成 UNIX 时 
间 惟 。 这 样 ,和 直接 调用 time() 函 数 获取 当前 的 UNIX 时 间 戳 功能 相同 。 参 数 也 可 以 从 
右 向 左 省 略 ,任何 省 略 的 参数 会 被 设置 成 本 地 日 期 和 时 间 的 当前 值 。 如 果 只 想 转变 日 期 ， 
对 具体 的 时 间 不 在 乎 ,可 以 将 前 三 个 转变 时 间 的 参数 都 设置 为 0。mktime() 函数 对 于 日 
期 运算 和 验证 非常 有 用 , 它 可 以 自动 校正 越界 的 输入 。 

2. 日 期 的 计算 

在 PHP 中 ,计算 两 个 日 期 之 间 相隔 的 长 度 ,最 简单 的 方法 就 是 通过 计算 两 个 UNIX 
时 间 戳 之 差 来 获得 。 

下 例 脚 本 5-5, 在 PHP 脚本 中 接收 来 自 HTML 表单 用 户 提交 的 出 生日 期 ,计算 这 个 
用 户 的 年 龄 。 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-5) 。 


脚本 5-5 利用 UNIX 时 间 改 计算 用 户 的 年 龄 。 


1 < !DOCTYPE html PUBLIC“"- //W3C//DTD XHTML 1.0 Transitional//EN™” 
2 "http://www.w3.0rg/TR/xhtml1/DID/xhtmll- transitional.dtd"> 
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3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
6 ”<title> 计 算 年 龄 < /title> 

7 </head> 

8 <body> 

9 


10 <?php #Script 5- 5-mktime.php 
11 $year=1985; 

12 $month=11; 

13 $day=13; 


15 $birthday=mktime(0, 0, 0, $month, $day, $ year); 
16 $nowdate=mktime(); 

17 $ageunix=$nowdate - $birthday; 

18 $age=floor($ageunix /(60* 60* 24* 365))7 

19 ”echo "年 龄 :$age"; 


21 ?> 
22 </body> 
23 </html> 


(2) 初始 化 用 户 的 生日 日 期 ,分 别 保存 在 $ year、$ month 和 $day 三 个 变量 中 ,这 里 


将 用 户 的 生日 1985 年 11 月 13 日 的 年 月 日 数值 直接 赋值 给 了 变量 ,在 后 面 章节 中 将 介绍 
如 何 从 表单 中 获取 用 户 的 输入 。 


(3) 将 出 生日 期 转变 为 UNIX 时 间 戳 ,并 保存 在 $ birthday 变量 中 。 
$birthday=mktime (0, 0, 0, $month, $day, $ year) 7 

(4) 获取 当前 时 间 的 UNIX 时 间 戳 ,并 保存 在 $nowdate 变量 中 。 
$nowdate=mktime ()7 

(5) 两 个 时 间 戳 相 减 获取 用 户 年 龄 的 UNIX 时 间 戳 .并 保存 在 $ageunix 变量 中 。 
$ageunix= Snowdate -Sbirthdayy 


(6) 将 UNIX 时 间 截 除 以 一 年 的 秒 数 获取 用 户 年 龄 ,计算 得 到 结果 为 28( 本 书 编写 


之 年 为 2014 年 ) ,并 保存 至 $age 中 ,然后 输出 。 


$age=floor ($ageunix /(60x 60* 24* 365)); 

echo "年 龄 :$ age”"; 

这 里 使 用 了 floor() 函 数 , 向 下 舍 入 为 最 接近 的 整数 。 

(7) 完成 脚本 ,将 文件 另存 为 mktime. php ,存放 在 Web 目录 中 ,并 在 Web 浏览 器 中 


测试 它 (参见 图 5-6) 。 


在 以 上 脚本 中 ,调用 mktime() 函 数 将 从 用 户 出 生日 期 转变 为 UNIX 时 间 戳 ,再 调用 


(DOB /ean p - ox |G Hi 


图 5-6 利用 UNIX 时 间 惟 计算 用 户 的 年 龄 


time() 本 数 获取 当前 时 间 的 UNIX 时 间 戳 。 因 为 这 个 日 期 的 格式 都 是 使 用 整数 表示 的 ， 
所 以 可 以 将 它们 相 减 ,又 将 计算 后 获取 的 UNIX 时 间 戳 除 以 一 年 的 秒 数 ,将 UNIX 时 间 


截 转变 为 


以 年 度量 的 单位 。 


3. 在 PHP 中 获取 日 期 和 时 间 

PHP 提供 了 多 种 获取 时 间 和 日 期 的 函数 ,除了 通过 mktime() 函 数 获 取 当 前 的 
UNIX 时 间 戳 外 ,还 可 调用 getdate() 函数 确 定 当 前 时 间 。 

getdate() 函 数 返 回 一 个 由 时 间 戳 组 成 的 联合 数组 ,参数 需要 一 个 可 选 的 UNIX 时 间 
戳 。 如 果 没有 给 出 时 间 戳 , 则 认为 是 当前 本 地 时 间 。 总 共 返 回 11 个 数组 元 素 , 如 表 5-1 


所 示 。 
表 5-1 getdate() 函 数 返 回 的 数组 单元 
键 名 描 述 返回 值 例 子 
hours 小 时 的 数值 表示 0 一 23 
mday 月 份 中 日 的 数值 表示 1~31 
minutes “| 分 钟 的 数值 表示 0 一 59 
mon 月 份 的 数值 表示 1 一 12 
month 月 份 的 英文 完整 文本 表示 January 一 December 
seconds 秒 的 数值 表示 0~59 
wday 一 周 中 日 的 数值 表示 0 一 6(0 表示 星期 日 ) 
weekday ”| 一 周 中 日 的 英文 完整 文本 表示 | Sunday 一 Saturday 
yday 一 年 中 日 的 数值 偏 移 0 一 365 
year 年 份 的 4 位 表示 例如 ,1999 或 2009 
0 UNIX 时 间 截 系统 相关 ,典型 值 为 从 一 2 147 483 648 一 2 147 483 647 


如 果 将 “2009 年 10 月 1 日 ,07:30:50 EDT” 转 变 为 UNIX 时 间 惟 1254382250 表示 ， 
并 将 其 传 给 getdate() 函 数 ,查看 各 数组 元 素 如 下 : 


Array( 


[seconds]=>50, 


[minutes]=>30, 


// 秒 的 数值 表示 
// 分 钟 的 数值 表示 


92 


PHP+MySQL 项 目 实例 开发 


[hours]=>7, // 小 时 的 数值 表示 

[mday]=>1, // 月 份 中 日 的 数值 表示 
[wday]=>4, // 一 周 中 日 的 数值 表示 
[mon]=>10, // 月 份 的 数值 表示 
[year]=>2009, // 年 份 的 4 位 表示 

[yday]=>273, // 一 年 中 日 的 数值 偏 移 
[weekday]=> Thursday, // 一 周 中 日 的 完整 文本 表示 
[month]=> October, // 月 份 的 完整 文本 表示 
[0]=>1254382250 // 自 从 UNIX 纪 元 开始 至 今 的 秒 数 


) 


4. 日 期 和 时 间 格 式 化 输出 

当日 期 和 时 间 需 要 保存 或 计算 的 时 候 , 应 该 使 用 UNIX 时 间 戳 作为 标准 格式 ,这 可 
以 作为 一 条 重要 的 规则 。 但 UNIX 时 间 戳 的 格式 可 读 性 比较 差 ,所 以 把 时 间 改 格 式 化 为 
可 读 性 更 好 的 日 期 和 时 间 ,或 格式 化 为 其 他 软件 需要 的 格式 。 在 PHP 中 可 以 调用 date() 
也 数 格式 化 一 个 本 地 时 间 和 日 期 ,该 函数 的 语法 如 下 : 


date (string format [, int timestamp]); 


date() 用 户 格 式 化 一 个 本 地 时 间 和 日 期 ,该 函数 有 两 个 参数 : 第 一 个 参数 是 必需 的 ， 
规定 时 间 戳 的 转换 格式 ;第 二 个 参数 是 可 选 的 ,需要 提供 一 个 UNIX 时 间 惟 。 如 果 没 有 
指定 这 个 UNIX 时 间 改 ,默认 值 为 mktime() 将 返回 当前 日 期 和 时 间 的 UNIX 时 间 戳 。 
该 函数 将 返回 一 个 格式 化 后 表示 适当 日 期 的 字符 串 。 

date() 函数 中 的 第 一 个 参数 ,是 通过 表 5-2 中 所 提供 的 特定 字符 组 成 的 格式 化 字符 
串 。 如 果 在 格式 字 串 中 的 字符 前 加 上 反 斜 线 来 转 义 可 以 避免 它 被 按照 表 5-2 解释 。 如 果 
加 上 反 和 斜 线 后 的 字符 本 身 就 是 一 个 特殊 序列 , 那 还 要 转 义 反 斜 线 。 格 式 字 串 中 不 能 被 识 
别 的 字符 将 原样 显示 。 表 5-2 给 出 了 PHP 中 所 支持 的 常用 日 期 格式 代码 ,更 多 参数 格式 
请 参考 PHP 帮助 手册 。 


表 5-2 PHP 的 date() 函 数 所 支持 的 格式 代码 


格式 化 字符 描 述 示 例 
Y 用 4 位 数字 表示 年 2011 
了 用 2 位 数字 表示 年 04 
n 用 1 位 或 2 位 数字 表示 月 份 3 
m 用 2 位 数字 表示 月 份 03 
F 月 份 的 英文 完整 文本 表示 February 
M 用 3 个 字母 表示 月 份 Feb 
ji 用 1 位 或 2 位 数字 表示 一 月 中 的 某 一 天 5 
d 用 2 位 数字 表示 一 月 中 的 某 一 天 05 
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续 表 
格式 化 字符 描 述 示 例 

1 一 周 中 日 的 英文 完整 文本 表示 (小 写 L) Monday 
D 用 3 个 字母 表示 星期 几 Mon 
g 小 时 ,用 1 位 或 2 位 数字 表示 的 12 小 时 格式 6 
G 小 时 ,用 1 位 或 2 位 数字 表示 的 24 小 时 格式 18 
h 小 时 ,用 2 位 数字 表示 的 12 小 时 格式 06 
H 小 时 ,用 2 位 数字 表示 的 24 小 时 格式 18 

分 45 
s 秒 56 
a am 或 pm am 
A AM 或 PM AM 


date() 函 数 可 以 带 有 这 些 参数 的 任意 组 合 , 来 确定 其 返回 的 结果 。 例 如 : 


echo date("Y 年 m 月 d 日 H:i:s"); 
// 输 出 当前 的 时 间 格 式 :2014 年 10 月 01 日 08:28:15 


5. 修改 PHP 的 默认 时 区 

每 个 地 区 都 有 自己 的 本 地 时 间 ,在 网 上 以 及 无 线 电 通信 中 ,时 间 的 转换 问题 就 显得 格 
外 突出 。 整 个 地 球 分 为 24 个 时 区 ,每 个 时 区 都 有 自己 的 本 地 时 间 。 在 国际 无 线 电 或 网 络 
通信 场合 ,为 了 统一 起 见 ,使 用 一 个 统一 的 时 间 , 称 为 通用 协调 时 (Universal Time 
Coordinated,UTC) ,是 由 世界 时 间 标 准 设 定 的 全 球 标准 时 间 。UTC 原先 也 被 称 为 格林 
尼 治 标准 时 间 (Greenwich Mean Time,GMT) ,与 英国 伦敦 的 本 地 时 间 相 同 。 

PHP 默认 的 时 区 设置 是 UTC 时 间 ,而 北京 正好 位 于 时 区 的 东 八 区 ,领先 UTC 八 个 
小 时 。 所 以 在 使 用 PHP 中 像 mktime() 等 获取 当前 时 间 的 函数 时 ,得 到 的 时 间 总 是 不 对 ， 
表现 是 和 北京 时 间 相差 八 个 小 时 。 如 果 和 希望 正确 显示 北京 时 间 ,就 需要 修改 默认 的 时 区 
设置 ,可 以 通过 以 下 两 种 方式 完成 。 

(1) 修改 php. ini 配置 文件 。 

如 果 使 用 的 是 独立 的 服务 器 .有 权限 修改 配置 文件 .设置 时 区 就 可 以 通过 修改 php 
.ini 中 的 date. timezone 属性 完成 。 可 以 将 这 个 属性 的 值 设 置 为 “Asia/Shanghai”、 
“Asia/Chongqing”、“Etc/GMT-8” 或 PRC 等 中 的 一 个 ,再 在 PHP 脚本 中 获取 的 当前 时 
间 就 是 北京 时 间 。 该 设置 需 重启 Web 服务 后 生效 。 

修改 PHP 的 配置 文件 如 下 所 示 : 


date.timezone=Asia/Shanghai; 


(2) 每 次 获取 时 间 前 设置 时 区 。 
如 果 使 用 的 是 共享 服务 器 ,没有 权限 修改 配置 文件 php. ini, 并 且 PHP 版 本 又 在 
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5.1.0 以 上 ,也 可 以 在 输出 时 间 之 前 调用 date_default_timezone_set() 函 数 设 置 时 区 。 该 
函数 需要 提供 一 个 时 区 标识 符 作 为 参数 ,和 配置 文件 中 date. timezone 属性 的 值 相同 。 
该 函数 的 使 用 如 下 所 示 : 


date default timezone set('PRC'); 


5.2.3 字符 串 处 理 函 数 


程序 处 理 过 程 中 离 不 开 对 数据 进行 存储 处理。 为 了 更 好 地 处 理 字符 ,我 们 将 任意 长 
度 的 字符 序列 定义 为 字符 串 。 字 符 串 可 以 存储 、 代 表 和 理解 实际 生活 中 很 多 的 数据 。 
PHP 提供 了 大 量 的 字符 串 处 理 函 数 , 帮 助 我 们 解决 实际 中 的 问题 。 以 下 介绍 几 个 常见 的 
PHP 字符 串 处 理 函 数 ,更 多 函数 请 参考 PHP 帮助 手册 。 

1. 获取 字符 串 长 度 

PHP 程序 开发 过 程 中 ,经 常 需 要 获取 某 些 特殊 字符 串 的 长 度 ,例如 在 用 户 注册 页 面 
中 ,可 以 通过 判断 用 户 输入 注册 信息 的 长 度 , 来 判断 该 注册 信息 是 否 完整 地 存储 到 数据 库 
中 ,因为 有 些 用 户 填写 的 注册 信息 可 能 超出 数据 库 中 存储 该 字段 的 长 度 ,导致 这 些 信息 只 
能 部 分 存储 。 

可 以 使 用 函数 strlen() 来 检查 字符 串 的 长 度 ,如 果 客 户 端 传 递 过 来 一 个 字符 串 ,我 们 
有 时 需要 对 字符 串 的 长 度 进行 限制 ,可 以 用 strlen() 返 回 字符 串 的 长 度 。 

语法 如 下 : 

strlen (string string); 


函数 的 返回 值 为 整 型 ,返回 字符 串 的 长 度 ,参数 为 字符 串 类 型 。 例 如 ; 


$str= "php'; 

echo strlen($ str); // 输 出 长 度 为 3 

注意 : 一 个 汉字 所 占 的 长 度 为 2。 

2. 比较 

运算 符 “ 二 = 二 ”可 以 用 来 比较 两 个 字符 串 是 否 相等 ,但 是 详细 的 比较 需要 通过 函数 来 
处 理 。PHP 中 对 字符 串 的 大 小 比较 可 以 通过 函数 strecasecmp() 和 函数 stremp() 来 实现 。 
语法 格式 如 下 : 

strcasecmp (stringl, string?2) 

strcmp (stringl, string2) 

这 两 个 函数 用 于 比较 两 个 字符 串 的 大 小 (基于 ASCII 码 值 ), 如 果 string1 二 string2， 
则 返回 1; 如 果 stringl 二 string2, 则 返回 一 1; 如果 stringl1 二 string2, 则 返回 0。 二 者 的 区 
别 在 于 函数 strcasecmp(0) 在 比较 的 过 程 中 不 区 分 大 小 写 ,而 函数 stremp() 在 比较 的 过 程 
中 区 分 大 小 写 。 例 如 : 


echo strcmp ("Hello world!", "Hello world!"); // 输 出 0, 代 表 两 个 字符 串 相等 
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echo strcasecmp ("Hello world!", "HELLO WORLD!"); // 输 出 0, 不 区 分 大 小 写 


3. 查找 替换 

在 Web 开发 过 程 中 ,如 果 对 数据 库 中 的 信息 处 理 不 当 , 可 能 导致 这 些 信息 无 法 原样 
输出 ,例如 在 数据 库 中 提取 字符 二”, 该 字符 无 法 正常 原样 输出 ,这 是 因为 “二” 为 HTML 
标记 的 一 部 分 ,如 “一 br>> 一 tr>、 一 td>” 等 标记 都 含有 “一 ”, 为 了 解决 上 述 问题 ,可 以 
设法 将 “二 ”替换 成 “&1t”, 因 为 在 HTML 标记 输出 时 ,将 *&lte" 识 别 为 "一 ”。 


1) 取 子 串 
如 果 需 要 从 字符 串 中 查找 ,取出 字符 ,可 以 使 用 substr() 函 数 , 它 可 以 返回 字符 串 的 
一 部 分 。 语 法 如 下 : 


substr (string, start, length); 


参数 string 是 必需 的 ,规定 要 返回 其 中 一 部 分 的 字符 串 。 参 数 start 也 是 必需 的 , 规 
定 在 字符 串 的 何 处 开始 。 

start 如 果 是 正 数 , 则 表示 字符 串 指定 的 开始 位 置 ; 如 果 是 负数 , 则 表示 从 字符 串 结尾 
开始 的 指定 位 置 ; 如 果 是 0, 则 表示 从 字符 串 的 第 一 个 字符 处 开始 。 

参数 length 可 选 ,规定 要 返回 的 字符 串 长 度 。 默 认 是 直到 字符 串 的 结尾 。 若 为 正 
数 , 则 表示 从 start 参数 所 在 的 位 置 返回 ; 若 为 负数 , 则 表示 从 字符 串 末端 返回 。 

substr() 函数 的 用 法 为 从 string 中 取出 字符 串 , 从 start 位 置 开 始 取出 length 的 长 
度 。 字 符 串 位 置 开始 值 为 零 。 如 果 没 有 指出 length, 那 么 默认 一 直到 字符 串 未 尾 。 

例如 : 


echo substr ("Hello world!", 6); // 输 出 :world! 


表示 从 “Hello world!1" 字 符 串 的 第 6 个 字符 开始 取 值 ,一 直 取 到 字符 串 结 尾 。 


echo substr ("Hello world!", 6,5); // 输 出 :world 

表示 从 “Hello world!1” 字 符 串 的 第 6 个 字符 开始 取 值 , 取 长 度 为 5 的 子 串 。 
下 例 脚 本 5-6, 在 PHP 脚本 中 将 数字 转换 为 图 片 显示 的 计数 器 效果 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-6) 。 


脚本 5-6 利用 取 子 串 函数 实现 图 片 效果 的 计数 器 。 


1 
2 
4 
5 
6 
7 
8 
9 


< !DOCTYPE html PUBLIC "- //W3C//DITD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- TYpe"” content= "text/html; charset=gb2312" /> 
<title> 计 算 年 龄 < /title> 
< /head> 
<body> 
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10 <?php #Script 5- 6- countimg.php 
11 $number=123456; 
12 $str=""; 


14 for($i=0;$i<strlen($ number);$i++){ 

15 $digit= substr ($ number, $1,1);} 

16 $str .="<img sro=\"img/$digit.gif\">"; 
17 )} 


3 2%> 

20 <p class="stylel">epple 的 减肥 营 < /p> 

21 <P class="style2"> 开 站 至 今 ,已 有 <?php echo $str?> 人 到 访 ,< /p> 

22 </body> 

23 </htm> 

(2) 初始 化 计数 器 的 值 ,存放 在 $ number 变量 中 。 这 里 直接 给 $ number 赋值 ,在 后 
面 章 节 中 将 介绍 如 何 从 文件 中 读 取 数 值 ,以 及 如 何 进行 动态 计数 。 

$ number=123456; 

(3) 初始 化 $ str 变量 ,存放 显示 数字 图 片 的 路 径 , 开 始 为 空 。 

$str=""; 

(4) 通过 for 循环 , 按 位 数 逐 个 读 取 $number 的 每 个 数值 ,并 将 读 出 来 的 每 个 数字 替 
换 为 图 片 显示 的 路 径 。 图 片 保存 在 当前 目录 下 的 img 文件 夹 , 且 每 张 图 片 都 以 这 张 图 片 
对 应 的 数字 命名 。 

for($i=0;$i <strlen($number);$ i++){ 

$digit= substr ($ number, $i,1); 
$str .="<img src=\"img/$ digit .gif\">";» 
} 

(5) 输出 图 片 ,调整 显示 CSS 样式 ,完成 脚本 ,将 文件 另存 为 countimg. php ,存放 在 
Web 目录 中 ,并 在 Web 浏览 器 中 测试 它 (参见 图 5-7) 。 

2) 替换 

PHP 中 实现 字符 串 的 替换 方法 有 多 种 .但 最 简单 的 方法 可 以 通过 函数 str_replace() 
实现 。 该 函数 的 语法 格式 如 下 : 


str_replace (stringl, string2, string) 


功能 : 该 函数 将 字符 串 string 中 的 子 串 stringl 替换 成 字符 串 string2 后 返回 新 字 
符 串 。 
str_replace() 函数 使 用 一 个 字符 串 替 换 字符 串 中 的 另 一 些 字符 。 例 如 : 


echo str_replace ("world", "John", "Hello world!"); // 输 出 :Hello John! 


另外 .在 处 理 HTML 字符 时 ,经 常 需要 调整 HTML 的 显示 效果 ,由 于 代码 中 的 空白 
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EE 


@pple 的 减肥 营 


开 站 至 今 ， 己 有 ISOGG 人 到 访 ， 


5-7 “利用 取 子 串 函数 实现 数字 显示 为 图 片 的 计数 器 效果 


与 显示 的 空白 是 不 同 的 ,所 以 需要 额外 处 理 空格 和 换行 等 效果 。 

以 下 是 常见 的 用 法 : 

5$new_content= str replace(" ","&nbsp;",$ content); 

$new_content= str_ replace("\n","<br />",$ new content); 

将 用 户 输入 的 空格 转换 为 HTML 显示 的 空格 特殊 符 ,将 用 户 输入 的 换行 转换 为 
HTML 显示 的 换行 符 。 

4. 删除 空白 

在 用 户 信 息 注 册页 面 或 商品 信息 添加 页 面 中 ,经 常 需要 对 用 户 录入 的 信息 进行 特殊 
处 理 , 例 如 录入 信息 统一 格式 的 转换 、 录 入 信息 首尾 空格 的 处 理 。 

trim() 函 数 从 字符 串 的 两 端 删除 空白 字符 和 其 他 预定 义 字 符 。 语 法 如 下 : 


trim(string, charlist); 


string 参数 是 必需 的 ,为 规定 要 检查 的 字符 串 。 
Charlist 参数 是 可 选 的 ,为 规定 要 转换 的 字符 串 。 如 果 省 略 该 参数 , 则 删除 以 下 所 有 


字符 : 
“ \M0"——NULL, 
a tab。 
new line。 
。"\x0B" 一 一 纵向 列表 符 。 
“EN 回 车 。 
。" "普通 空白 字符 。 


另外 ,ltrim()(1 是 指 left) 只 删除 字符 串 开头 的 空白 。rtrim()(r 是 指 right) 只 删除 字 
符 串 末尾 的 空白 。 用 法 与 trim() 函 数 相 同 。 例 如 : 


$title=" Programming PHP \n"; 
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$str 1=1trim($title); //$str 1 是 "Programming PHP \n" 
$str 2=rtrim($ title); //Sstr 2 是 " Programming PHP" 
$str 3=trim($title); //Sstr 3 是 "Programming PHP" 


5.3 项 目 州 练 


随机 数 验 证 码 图 片 的 制作 


5.3.1 项 目 背景 与 思路 


PHP 不 仅 可 以 生成 HTML 页 面 ,而 且 可 以 创建 和 操作 二 进 制 形式 的 数据 ,如 图 像 、 
文件 等 。 

1. 了 解 GD 函数 库 

使 用 PHP 操纵 图 形 可 以 通过 GD2 函数 库 来 实现 ,利用 GD2 函数 库 可 以 在 页 面 中 绘 
制 各 种 图 像 .统计 图 。GD2 库 是 一 个 开放 的 ,动态 创建 图 像 的 源 代码 公开 的 函数 库 ,可 以 
从 官方 网 站 http://www. boutell. com/gd 下 载 最 新 版 本 的 GD2 函数 库 。 目 前 ,GD2 函 
数 库 支持 gif ,png、jpeg、wbmp 和 xbm 等 多 种 图 像 格 式 。 

GD2 函数 库 是 一 个 源 代码 公开 的 用 于 创建 图 形 图 像 的 函数 库 , 该 函数 库 由 C 语言 编 
写 ,可 以 在 Perl`PHP 等 多 种 语言 中 调试 运用 。PHP 使 用 GD2 函数 库 可 以 制作 出 各 类 让 
富 的 图 形 图 像 效 果 ,如 统计 图 ,为 图 片 添加 水 印 以 及 动态 图 表 等 ,在 企业 OA 管理 系统 、 电 
子 商务 网 站 等 大 型 项 目 中 得 以 广泛 应 用 。 很 多 开源 项 目 都 对 GD2 函数 库 提供 了 很 好 的 
技术 支持 ,如 Jpgraph 类 库 就 是 基于 GD2 函数 库 开发 的 用 于 制作 复杂 统计 图 的 类 库 。 

在 使 用 基本 的 图 像 创建 函数 之 前 ,需要 安装 GD 库 文件 。 如 果 要 使 用 与 JPEG 有 关 
的 图 像 创建 函数 ,还 需要 安装 jpeg-6b; 如 果 要 在 图 像 中 使 用 Typel 型 字体 , 则 必须 安 
装 tllib。 

PHP 5 中 GD2 函数 库 已 经 作为 扩展 被 默认 安装 ,但 目前 有 些 版 本 中 ,还 需要 对 php 
.ini 文件 进行 设置 来 激活 GD2 函数 库 。 

。 若是 Windows 环境 , 则 将 “; extension 二 php_gd2. dll” 选 项 中 的 分 号 ";” 删 除 , 如 

图 5-8 所 示 。 

。 若是 Linux 环境 , 则 将 “ ;extension 二 php_gd2. so” 选 项 中 的 分 号 ";” 删 除 。 

保存 修改 后 的 文件 ,并 重新 启动 Apache 服务 器 即 可 激活 GD2 函数 库 。 

在 成 功 加 载 GD2 函数 库 后 ,可 以 通过 phpinfo() 函数 来 获取 GD2 函数 库 的 安装 信 
息 ,验证 GD 库 是 否 安装 成 功 。 如 果 在 打开 的 页 面 中 检索 到 的 GD 库 的 安装 信息 ,如 图 5-9 
所 示 , 说 明 GD 库 安 装 成 功 ,这 样 开发 人 员 就 可 以 在 程序 中 使 用 GD2 函数 库 编 写 图 形 图 像 。 

2. 图 像 处 理 函 数 

在 PHP 中 处 理 一 个 图 像 应 该 完成 如 下 所 示 的 四 个 步 又 : 

(1) 创建 一 个 背景 图 像 , 后 继 操 作 都 基于 此 背景 图 像 。 

(2) 在 背景 上 绘制 图 像 轮廓 或 输入 文本 。 

(3) 输出 最 终 图 像 。 
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579 ;extension=php_mbstring.dll 

580 ;extension=php_bz2.d1l 

581 ;extension=php_curl.dll 

58z ;extension=php_dba.dll 

583 extension=php_dbase.dll 

584 ;extension=php_exif.dll 

595 ;extension=php_fdf.dll 

586 ;extension=php filepro.dll 

587 
:extension=php_gettext.dll 
;extension=php_ifx.dll 
:extension=php_imap.dll 
:extension=php_interbase.dll 
:extension=php_ldap.dll 

593 extension=php_mbstring.dll 

594 ;extension=php_merypt.dll 


图 5-8 激活 GD2 函数 库 


phpinfo() - Vindows Internet Erplorer =| 口 |x| 


@GiSDET7ZETETTTTTT3TTTPD | 网 | CR 站 日 


[TT me [om - 
图 5-9 GD2 函数 库 的 安装 信息 


(4) 释放 资源 。 

PHP 中 有 一 组 图 像 函数 ,可 以 动态 生成 各 种 格式 的 图 像 数 据 流 并 输出 到 服务 器 。 为 
了 这 组 函数 能 够 工作 ,系统 中 必须 有 GD 函数 库 的 支持 。 更 多 图 像 处 理 函 数 请 参考 PHP 
帮助 手册 与 GD 库 帮 助手 册 。 
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常用 的 图 像 函数 有 以 下 几 个。 

1) imageCreate 

GD2 函数 库 在 图 形 图 像 绘制 方面 功能 非常 强大 ,开发 人 员 既 可 以 在 已 有 图 片 的 基础 
上 进行 绘制 ,也 可 以 在 没有 任何 素材 的 基础 上 绘制 。 在 这 种 情况 下 首先 要 创建 画布 ,之 后 
所 有 操作 都 将 依据 所 创建 的 画布 进行 。 在 GD2 函数 库 中 创建 画布 通过 imagecreate() 函 
数 实现 。 语 法 如 下 : 


imagecreate (int x size, int y size); 


imagecreate( ) 函数 用 来 建立 一 张 全 空 的 图 形 。 参 数 x_size、y_size 为 图 形 的 尺寸 , 单 
位 为 像素 (pixel) 。 

2) imageCreateFromPNG 

车 在 已 有 图 片 的 基础 上 进行 绘制 就 可 使 用 该 函数 ,取出 PNG 图 形 ,作为 图 像 处 理 的 
背景 。 语 法 如 下 : 


imagecreatefrompng (string filename); 


imagecreatefrompng() 函 数 用 来 取出 一 张 PNG 格式 图 形 , 通 当 取出 当 背 景 或 者 基本 
的 画布 样本 使 用 。 参 数 filename 可 以 是 本 地 文件 路 径 ,也 可 以 是 网 络 URL 地 址 。 
ImageCreateFromPNG 返回 值 为 PNG 的 文件 代码 ,可 供 其 他 的 函数 使 用 。 本 函数 在 
PHP 3. 0. 13 版 之 后 才 支 持 。 

3) imageColorAllocate 


该 函数 用 于 分 配 一 个 调 色 板 项 ,匹配 颜色 。 语 法 如 下 : 
imagecolorallocate (int im, int red, int green, int blue); 


imageColorAllocate() 函 数 用 来 匹配 图 形 的 颜色 . 供 其 他 绘图 函数 使 用 。 参 数 im 表 
示 图 形 的 图 像 描述 符 。 参 数 red .green、blue 是 色彩 三 原色 ,其 值 为 0 一 255 。 

4) imageTTFBBox 

该 函数 用 于 计算 TTF 文字 所 占 区 域 。 语 法 如 下 : 


imageTTFBBox (int size, int angle, string fontfile, string text); 


imageTTFBBox() 函 数 用 来 计算 并 返回 TTF 文字 区 域 框 大 小 。 参 数 size 为 字 型 的 
尺寸 ;angle 为 字 型 的 角度 ;fontfile 为 字体 文件 名 称 , 也 可 以 是 远程 文件 ;text 是 字符 串 内 
容 了 。 

imageTTFBBox 的 返回 值 为 数组 ,包括 了 八 个 元 素 ,前 二 个 分 别 为 左下 的 x、y 坐标 ， 
第 三 、 四 个 为 右 下 角 的 x、y 坐标 ,第 五 六 及 七 、 八 二 组 分 别 为 右上 及 左上 的 x、y 坐标 。 

5) imagesx 

该 函数 用 于 取得 图 片 的 宽度 。 语 法 如 下 : 

imagesx (int im); 


imagesx() 函 数 用 来 取得 图 片 的 宽度 数值 。 
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6) imagesy 

该 函数 用 于 取得 图 片 的 高 度 。 语 法 如 下 : 

imagesy (int im); 

imagesy() 函数 用 来 取得 图 片 的 高 度数 值 。 

7) imagettftext 

imagettftext() 函数 的 应 用 可 以 将 TTF 格式 的 文字 放 入 图 片 中 ,在 实际 的 代码 编程 
处 理 图 片 中 是 非常 有 用 的 一 个 函数 。 语 法 如 下 : 

imagettftext (int im, int size, int angle, int x, int y, int col, string fontfile, string 

text); 

imagettftext() 函数 将 ttf(TrueType fonts) 字 体 文字 写 和 图片。 参数 size 为 字形 的 
尺寸 ;angle 为 字 型 的 角度 , 顺 时 针 计算 ,0 度 为 水 平 ,也 就 是 三 点 钟 的 方向 (由 左 到 右 )， 
90 度 则 为 由 下 到 上 的 文字 ;x 和 y 二 参数 为 文字 的 坐标 值 (原点 为 左上 角 )。 

参数 col 为 字 的 颜色 ;fontfile 为 字体 文件 名 称 , 也 可 以 是 远程 文件 ;text 是 字符 串 内 
容 了 。imagettftext() 返 回 一 个 含有 8 个 元 素 的 数组 表示 了 文本 外 框 的 4 个 角 , 顺 序 为 左 
下 角 、 右 下 角 、 右 上 角 、 左 上 角 。 这 些 点 是 相对 于 文本 的 而 和 角度 无 关 , 因 此 “左上 角 ” 指 的 
是 以 水 平方 向 看 文字 时 的 左上 角 。 

8) imagePNG 

该 函数 用 于 建立 PNG 图 形 。 语 法 如 下 : 

imagepng (int im, string [filename]); 

imagePNG() 函 数 用 来 建立 一 张 PNG 格式 图 形 。 参 数 im 为 使 用 imageCreate() 所 
建立 的 图 像 描 述 符 。 参 数 filename 可 省 略 , 若 省 略 , 则 会 将 图 片 直接 送 到 浏览 器 端 ,记得 
在 送出 图 片 之 前 要 先 使 用 header 方法 送出 页 面 的 MIME 类 型 :“Contentrtype: image/ 
png”, 以 顺利 传输 图 片 。 本 函数 在 PHP 3. 0. 13 版 之 后 才 支持 。 

9) imageDestroy 

该 函数 用 于 结束 图 形 。 语 法 如 下 : 

imagedestroy (int im); 

imageDestroy() 函 数 将 图 像 描述 符 解 构 , 释 于 内 存 空 间 。 参 数 im 为 imageCreate() 
所 建立 的 图 像 描述 符 。 


5.3.2 设计 过 程 


1. 生成 图 像 

脚本 5-4 完成 了 如 何 生成 随机 字符 串 , 这 里 将 实现 如 何 把 生成 的 随机 字符 串 以 图 片 
的 形式 显示 。 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 5-7) 。 


101 


102 PHP+MySQL 项 目 实例 开发 


脚本 5-7 利用 图 像 处 理 函 数 实现 将 文字 显示 在 图 片上 。 


wm PP 
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<?php 
$length=5; 
$string="'; 
$arr=array(a,b,c,d,e,f,g,h,i,js k,lmn,o pd, r,s tu VW, XY, ZA B,C,D, EF,G,H, I, 
TL MNOPOQORS TUV LY21234,5,6:7,8,9.0)7 


$arrlength= count ($arr); 
for($i=0;$ i<$ length;$ i++) 
$string .=$arr[rand(0,$arrlength-— 1)]; 


$ font= 'Rrial'7 

$size=12; 

$ im= imageCreateFromPNG('button.png');} 

$ tsize= imagettfbbox ($ size, 0, $ font, $ string); 


$dx=abs ($tsize[2]- $tsize[0]); 
$dy=abs ($tsize[5]- $tsize[3]); 
$x= (imagesx ($ im)- $ dx) /2; 

$y= (imagesy ($ im)- $dy) /2+ $ dy; 


$black= imageColorAllocate ($ im, 0, 0, 0); 
TmageTTFText ($ im, $ size, 0, $x, $y, $black, $ font, $ string); 


header ('Content- type: image/png'); 
imageFNG ($ im) 7? 
imagedestroy ($ im); 


Et 


(2) 首先 生成 随机 字符 串 , 此 处 2 一 8 行 代码 即 脚本 5-4 的 部 分 代码 。 
(3) 初始 化 字体 文件 为 “Arial” ,字体 大 小 为 12 号 字 。 


$ font= 'Arial'; 


$size=12; 
(4) 使 用 背景 图 片 button. png, 这 里 需要 事先 制作 一 张 合适 大 小 的 PNG 格式 的 背景 
图 纹 图 片 。 


$ im- imageCreateFromPNG ('button.png'); 


(5) 计算 要 显示 的 随机 字符 串 所 占 区 域 。 


$ tsize= imagettfbbox ($ size, 0, $font, $string); 
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(6) 要 是 文字 居中 显示 ,需要 计算 文字 显示 位 置 。abs() 函 数 为 取 绝 对 值 的 函数 。 


$dx=abs ($ tsize[2]- $tsize[0]); 
$dy=abs($tsize[5]- $tsize[3]); 
$x= (imagesx($ im)- $dx) /2; 

$y= (imagesy ($ im)- $ dy) /2+ $ dy; 


(7) 使 用 黑色 作 图 。 

$black= imageColorAllocate ($ im, 0, 0, 0); 

(8) 将 文字 写 到 图 片上 。 

imageTTFText ($ im, $ size, 0, $x, $y, $black, $font, $string); 


(9) 将 图 像 直 接 输 出 到 浏览 器 ,需要 告诉 浏览 器 输出 的 是 一 个 图 像 ,而 不 是 文本 或 
HTML。 通 过 调用 header() 函 数 制定 图 像 的 MIME 类 型 。 关 于 header() 函数 的 使 用 ,会 
在 后 续 章节 详细 介绍 。 


header ('Content- type: image/png'); 
(10) 生成 图 片 。 

imagePNG(S im) ; 

(11) 释放 内 存 。 

imagedestroy ($ im) 7 


(12) 完成 脚本 ,将 文件 另存 为 randimage. php ,存放 在 Web 目录 中 ,并 在 Web 浏览 
器 中 测试 它 ( 参 见 图 5-10) 。 


~ 上 X|‖ 合 iocalhost 


图 5-10 随机 字符 显示 为 图 片 


该 页 面 每 刷新 一 次 ,就 会 产生 一 个 新 的 随机 字符 ,图 片上 的 文字 就 会 变化 一 次 。 

2. 使 用 生成 的 图 像 

脚本 5-7 结合 其 他 应 用 需求 ,能 够 完成 表单 注册 或 登录 的 验证 码 等 功能 。 这 段 代码 
只 可 单独 使 用 ,不 可 直接 嵌入 到 其 他 代码 中 。 若 要 在 其 他 页 面 中 显示 这 张 图 片 ,可 以 在 调 
用 imagePNG0 〇 函数 时 ,就 将 生成 的 图 片 输出 为 文件 ,再 使 用 图 片 文件 。 或 者 可 以 使 用 脚 
本 5-8 的 用 法 。 

下 例 脚 本 5-8, 使 用 PHP 生成 的 图 片 。 
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(1) 在 文本 编辑 器 中 创建 一 个 新 的 HTML 文档 (参见 脚本 5-8) 。 


脚本 5-8 在 网 页 中 显示 PHP 生成 的 图 片 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtmll- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title> 显 示 验 证 码 图 片 < /title> 
</head> 
<body> 


<form action="" method= "get"> 

请 输入 验证 码 :< input name= "confirm" type= "text" /> 
< img src= "randimage.php" /> 

< /form> 

< /body> 

</htm> 


(2) 创建 表单 ,添加 文本 框 控件 。 
(3) 在 需要 显示 PHP 图 片 时 ,使 用 img 标签 添加 图 片 文件 。 


< img src= "randimage.php" /> 

特别 注意 的 是 ,这 里 添加 的 图 片 路 径 就 是 脚本 5-7 的 PHP 文件 路 径 ,这 里 的 image 
.Php 是 一 张 图 。 

(4) 完成 页 面 ,因为 没有 PHP 代码 ,所 以 可 保存 为 HTML 文件 ,将 文件 另存 为 
showimage. html, 存 放 在 Web 目录 中 ,并 在 Web 浏览 器 中 测试 它 (参见 图 5-11) 。 


A Ge httpy/localhos/11 D ~ OX 


图 5-11 在 网 页 中 显示 PHP 生成 的 图 片 


本 章 小 结 


本 章 的 重点 是 如 何 定 义 并 使 用 用 户 自 定义 函数 ,主要 包括 函数 定义 的 一 般 形式 、 函 数 
的 参数 和 返回 值 , 并 讨论 了 变量 作用 域 在 函数 定义 中 的 作用 。 本 章 的 另 一 个 重点 是 对 
PHP 中 的 常用 函数 、 时 期 和 时 间 、 字 符 串 、 图 像 及 发 送 邮 件 函 数 做 了 详细 讲解 ,这 些 要 点 


第 5 章 函 数 
在 PHP 开发 中 有 着 极其 重要 的 地 位 ,是 深入 学 习 PHP 和 熟练 开发 的 基础 。 
重点 回顾 


1. 如 何 自 定 义 各 种 函数 ,包括 传递 参数 、 设 定 默认 参数 及 返回 值 。 
2. 各 种 应 用 函数 。 


本 章 实 训 
【 实 训 1 


新 建 date. php, 参 考 “PHP 中 文 帮助 ,修改 时 区 ,使 用 时 间 日 期 函数 库 , 输 出 当前 日 
期 .星期 和 时 间 。 网 页 输出 格式 如 图 5-12 所 示 。 


2014 年 3 月 5 日 星期 三 


现在 的 时 间 为 ，02:26:36 
现在 的 时 间 为 ，02 时 26 分 36 秒 


5-12 date. php 的 运行 效果 


【 实 训 2】 

修改 第 4 章 项 目 训练 checkfile. php, 重 命名 为 checkfunc, php, 将 检查 文件 后 级 名 的 
代码 修改 为 函数 ,并 分 别 检查 myfilel. doc、myfile2. jpg 和 myfile3. exe 三 个 文件 的 格式 
是 否 符合 要 求 。 网 页 运行 效果 如 图 5-13 所 示 。 


文件 名 为 ，myfilel. doc 
文件 myfilel. doc 的 格式 错误 ， 必 须 是 图 片 ! 


文件 名 为 ，myfile2. jpg _ 
文件 myfile2. jpg 的 格式 符合 要 求 ， 可 以 上 传 ! 


文件 名 为 ，myfile3. exe 
文件 myfile3. exe 的 格式 错误 ， 必 须 是 图 片 ! 


图 5-13 ”checkfunc. php 的 运行 效果 
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6.1 尖 TML 表单 


HTML 是 一 种 简单 的 标记 语言 ,为 使 用 者 提供 了 极 大 的 灵活 性 ,这 一 点 使 它 很 容易 
学 习 和 编写 ,也 同样 是 由 于 这 一 点 , 太 多 的 网 页 设计 人 员 对 HTML 的 设计 与 编码 的 滥用 
或 不 规范 ,导致 一 个 页 面 在 下、Firefox、Mozilla 几 个 不 同 浏览 器 中 显示 的 效果 千差万别 。 

如 今 的 Web 设计 已 经 启用 新 的 标准 , 旨 在 使 网 页 的 HTML 只 包含 内 容 和 信息 ,以 标 
准 HTML 和 CSS 存储 信息 的 方式 ,也 就 是 现在 流行 的 DIV 十 CSS 设计 标准 。 

有 一 些 人 建议 使 用 XML 来 取代 HTML 语言 。 虽 然 XML 有 许多 强大 的 功能 ,不 过 
因为 人 门 的 门槛 较 高 ,让 人 望 而 生 旦 ,而 且 目前 有 太 多 的 HTML 型 网 站 ,因此 目前 沿用 
的 标准 是 HTML 与 XML 的 兼容 规格 ,叫做 XHTML ,作为 从 HTML 到 XML 的 过 渡 。 
创建 和 处 理 表 单 是 PHP 开发 者 的 一 个 重要 能 力 指 标 。 


6.1.1 创建 HTML 表单 


利用 PHP 管理 HTML 表单 是 一 个 包含 两 个 步骤 的 过 程 : 首先 ,使 用 选择 的 任何 文 
本 编辑 器 或 所 见 即 所 得 编辑 器 创建 HTML 表单 自身 ;然后 ,创建 相应 的 PHP 脚本 ,用 于 
接收 和 处 理 表 单数 据 。 

HTML 表单 是 使 用 form 标签 和 多 种 输出 控件 类 型 创建 的 。form 标签 形式 如 下 : 


< form action= "script .php" method= "post"> 
< /form> 


form 标签 最 重要 的 属性 是 action, 它 指定 将 把 表单 数据 发 送 到 哪个 页 面 。action 标 
签 指 的 是 接收 处 理 结 果 的 文件 位 置 , 当 action 值 为 空 时 , 则 提交 给 当前 文件 本 身 ;如 果 
action 的 值 为 其 他 文件 或 URL, 则 提交 给 该 文件 或 URL 地 址 处 理 。 

第 二 个 属性 method 指定 如 何 把 数据 发 送 到 处 理 页 面 。 它 有 两 种 值 : get 和 post, 如 
果 没 有 设置 method 属性 或 该 属性 为 空 值 .浏览 器 默认 method 的 值 为 post 方法 。 

值得 一 提 的 是 ,如 果 在 HTML 中 缺少 表单 结束 标记 过 /form> ,那么 整个 表单 是 不 
会 触发 任何 提交 动作 的 。 在 实际 开发 时 ,可 能 会 发 现 单 击 按钮 没有 任何 反应 ,其实 细 心 检 
查 一 下 表单 的 代码 就 可 以 了 ,有 时 即使 少 写 了 一 个 HTML 字符 ,浏览 器 也 不 会 正常 工 
作 的 。 
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1. get 和 post 的 区 别 

表单 的 method 属性 指定 如 何 把 数据 发 送 到 处 理 页 面 。 两 个 选项 get 和 post 指示 要 
使 用 的 HTTP(Hypertext Transfer Protocol, 超 文本 传输 协议 ) 方 法 。 如 果 要 在 浏览 器 中 
发 送 表单 或 数据 给 服务 器 端 ,使 用 get 或 post 方法 都 能 实现 。 

get 方法 是 在 访问 URL 时 ,使 用 浏览 器 地 址 栏 来 传递 值 。get 方法 把 提交 的 数据 通 
过 一 系列 追加 到 URL 后 面 的 名 - 值 (name-value) 对 发 送 到 接收 页 面 。 可 以 在 很 多 网 站 上 
看 到 这 类 URL 串 ,例如 : 


WwW.Ssina.com.cn/script .php? name=Homer&gender=M 


使 用 get 方法 的 好 处 是 : 可 以 将 得 到 的 页 面 添加 到 收藏 夹 ,因为 它 是 一 个 URL。 因 
此 ,可 以 在 Web 浏览 器 中 单 击 “ 后 退 ” 按 钮 返回 到 一 个 get 页 面 ,或 者 重新 加 载 它 , 而 不 会 
有 任何 问题 。 而 对 于 post, 则 不 能 执行 这 两 种 操作 。 

get 方法 方便 直观 ,缺点 是 访问 该 网 站 的 用 户 也 可 以 修改 URL 串 后 发 送 给 服务 器 ， 
如 果 程 序 处 理 得 不 够 好 ,很 容易 出 错 ,而 且 get 传递 的 字符 串 长 度 不 能 超过 250 个 字符 ， 
如 果 超 长 ,浏览 器 会 自动 截断 ,导致 数据 缺失 。 另 外 ,get 方法 不 支持 ASCII 字符 之 外 的 
任何 字符 ,比如 包含 有 汉字 或 其 他 非 ASCII 字符 时 ,需要 使 用 额外 的 编码 操作 。 

post 方法 发 送 变量 数据 时 ,对 于 用 户 来 说 是 不 透明 的 , 按 HTTP 协议 来 说 ,数据 附加 
于 header 的 头 信息 中 ,用 户 不 能 随意 修改 ,这 对 于 Web 应 用 程序 而 言 , 安 全 性 要 好 得 多 ， 
而 且 使 用 post 可 以 发 送 大 容量 的 数据 给 Web 服务 器 。 

因为 post 是 随 HTTP 的 header 信息 一 起 发 送 的 , 当 触 发 post 表单 提交 后 ,如 果 在 
Web 浏览 器 中 单 击 “ 后 退 ” 按 钮 .浏览 器 不 会 自动 重 发 POST 数据 。 如 果 用 户 此 时 单 击 
“刷新 ”按钮 ,将 会 有 “数据 已 经 过 期 ,是 否 重新 提交 表单 ”的 提示 ,这 一 点 不 如 get 使 用 方 
便 。 使 用 get 传 值 时 ,即便 用 户 使 用 “后 退 ” 或 “刷新 ”按钮 .浏览 器 的 URL 地 址 也 是 仍然 
存在 的 。 

因此 ,在 开发 中 需要 根据 实际 应 用 灵活 选择 get 和 post 来 提交 表单 数据 。 在 本 书 中 
一 般 使 用 post, 如果 出 现 例 外 情况 ,会 另外 指出 。 在 http://www. w3. org/2001/tag/ 
doc/whenToUsGet, html 上 可 以 找到 一 份 稍 具 技术 性 的 讨论 ,其 中 指出 了 在 什么 时 候 应 
该 使 用 哪 种 方法 。 

2. 表单 元 素 

在 form 的 打开 标签 和 封闭 标签 内 可 以 放置 不 同 的 输入 ,它们 可 以 是 文本 框 . 单 选 按 
钮 选项 菜单 、 复 选 框 等 。 表单 所 使 用 的 控件 标签 元 素 有 十 几 个 ,PHP 开发 中 常用 及 较 重 
要 的 标签 如 表 6-1 所 示 。 


表 6-1 常用 表单 控件 


表单 元 素 说 明 
input type= "text" 单行 文本 框 
input type= "radio" 单 选 按钮 ,用 于 设置 一 组 选择 项 ,用 户 只 能 选择 一 个 
input type 二 "checkbox” | 复 选 框 ,允许 用 户 选择 多 个 选择 项 
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续 表 


表单 元 素 


说 明 


input type 一 "file" 


文件 浏览 框 , 当 文件 上 传 时 ,可 用 来 打开 一 个 模式 窗口 以 选择 文件 


input type 一 "password" 


密码 文本 框 ,用 户 在 该 文本 框 输入 字符 时 将 被 替换 显示 为 * 号 


input type 一 "hidden” 


隐藏 标签 ,用 于 在 表单 中 以 隐 含 方式 提交 变量 值 


input type= "submit" 


表单 提交 按钮 


input type= "reset" 


清除 与 重 置 表单 内 容 , 用 于 清除 表单 中 所 有 文本 框 的 内 容 , 而 且 使 选择 菜 
单项 恢复 到 初始 值 


select 


下 拉 列 表 框 ,可 单 选 和 多 选 。 默 认为 单 选 ,如 果 增 加 多 项 选择 功能 ,增加 
二 select name 一 "select” size 一 " 自 定 义 列 数 ”" multiple 一 "mnultiple" 之 即 可 


option 


列表 下 拉 菜单 ,和 select 配合 使 用 ,显示 供 选择 的 值 


textarea 


多 行文 本 框 , 在 使 用 文本 框 时 需要 关闭 标签 之 间 的 文本 内 容 , 形 成 如 下 格 
式 : 一 textarea> 你 的 文字 一 /textarea> 


其 中 ,password 密码 文本 框 用 于 隐藏 密码 ,用户 输入 的 文本 将 以 * 显示 在 文本 框 中 ， 
但 是 密码 并 没有 加 密 ,只 是 被 * 替换 显示 ,这 点 请 注意 。 

hidden 标签 被 称 为 隐藏 或 隐 含 的 标签 , 它 不 会 在 用 户 浏览 的 页 面 界 面 上 出 现 , 当 用 
户 填 写 资料 表单 和 跨 页 之 间 传 值 时 ,可 以 使 用 该 标签 传递 一 些 隐 含 的 值 。 

表单 的 属性 用 于 表单 中 约束 表单 元 素 的 行为 或 显示 ,其 含义 与 约束 如 表 6-2 所 示 。 


属性 名 称 


表 6-2 常用 属性 及 含义 
说 明 


name 控件 的 名 称 ,PHP 根据 该 名 称 ,在 超级 全 局 数组 中 建立 以 name 为 名 称 的 键 名 


value 


本 框 中 


控件 的 默认 值 ,注意 ,该 值 不 能 应 用 到 type 一 password 密码 文本 框 以 及 type 一 file 文件 文 


size 文本 框 的 宽度 ,在 select 下 拉 菜 单 中 ,表示 可 以 看 到 的 选项 行 数 


multiple | 此 属性 用 于 下 拉 列 表 菜 单 select 中 ,指定 该 选项 用 户 可 以 使 用 Ctrl 和 Shift 键 进 行 多 选 


Tows 多 行文 本 框 显示 时 可 以 容纳 的 字符 行 数 宽度 


cols 多 行文 本 框 显示 时 可 以 容纳 的 字符 列 数 高 度 


除了 以 上 一 些 必要 的 属性 元 素 外 ,还 有 一 些 标准 属性 ,如 class、style\id 等 ,可 以 参阅 
HTML 的 相关 资料 。 不 过 ,在 这 些 属性 中 ,最 重要 的 就 是 应 该 注意 为 表单 输入 提供 的 名 
称 (name 属性 ) ,因为 当 表单 遇 到 PHP 代码 时 ,这些 名 称 将 会 是 至 关 重 要 的 。 


3. 创建 表单 


(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 6-1) 。 


脚本 6-1 这 个 简单 的 HTML 表单 将 用 于 本 章 中 的 多 个 示例 。 


1 < !DOCTYPE html PUBLIC "- //W3C/V/VDTD XHTML 1.0 Transitional//EN™” 


上 waow me w 
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"http://www.w3.0rg/TR/xhtml1/DTD/xhtmll- transitional .dtd"> 
<html xmlns= "http://www.w3.0org/1999/xhtml"> 


<head> 
<meta http- equiv= "Content- Type”" content= "text/html; charset=gb2312" /> 
<title> 简 易 的 表单 < /title> 
</head> 
<body>< !-- Script 6-1- form.html- 一 > 


10 “form action= "handle form.php" method= "post"> 


11 <fieldset><legend> 请 输入 您 的 信息 :< /legend> 

12 <p><b> 姓 名 :< /b> 

13 <input type= "text" name= "name" size= "20" maxlength= "40" />< /p> 
14 <p><b> Email:< /b> 

15 <input type= "text" name= "email" size= "40" maxlength="60" />< /p> 
16 <p><b> 性 别 < /b> 

17 <input type= "radio" name= "gender" value=" 男 " /> 男 

18 <input type="radio" name= "gender" value=" 女 " /> 女 < /p> 

19 <p><b> 年 龄 < /b> 

20 < select name= "age"> 

21 <option value= "0- 29">30 岁 以 内 < /option> 

22 <option value="30- 60"> 30 岁 至 60 岁 < /option> 

23 <option value="60+"> 60 岁 以 上 < /option> 

24 </select>< /p> 

25 <p><b> 个 人 简介 < /b> 

26 <textarea name= "comments" rows="3" cols="40">< /textarea>< /p> 
27 </fieldset> 

28 <div align= "center"> 

29 <input type= "submit" name= "submit" value= "提交 " /> 

30 </div> 

31 </form> 

32 </body> 

33 </html> 


(2) 插入 初始 form 标签 。 


< form action= "handle form.php" method= "post"> 

因为 action 属性 指定 将 把 表单 数据 发 送 到 哪个 脚本 ,应 该 给 它 提供 一 个 合适 的 名 称 
(handle_form, 以 对 应 于 这 个 脚本 : form. html) 以 及 . php 扩展 名 ,因为 PHP 页面 将 处 理 
这 个 表单 的 数据 。 

(3) 开始 创建 HTML 表单 。 


<fieldset>< legend> 请 输入 您 的 信息 :< /legend> 


fieldset 和 legend 这 两 个 HTML 标签 ,用 它们 创建 的 HTML 表单 的 外 观 , 它 们 在 表 
单 周 围 添加 了 一 个 方 框 ,并 在 顶部 设置 了 一 个 标题 。 尽 管 这 与 表单 自身 是 不 相关 的 。 
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(4) 添加 两 个 单行 文本 框 。 


<p><b> 姓 名 :< /b> 

< input type= "text" name= "name" size="20" maxlength="40" />< /p> 

<p><b>Email:< /b> 

< input type= "text" name= "email" size="40" maxlength= "60" />< /p> 

它们 只 是 简单 的 文本 输入 框 ,允许 用 户 输入 他 们 的 名 称 和 电子 邮件 地 址 。 每 个 输入 
框 标签 未 尾 额 外 的 空格 和 和 斜 村 是 有 效 的 XHTML。 例 如 ,利用 标准 的 HTML, 作 为 替代 ， 
这 些 标 签 可 以 用 maxlength 二 "40" 志 或 maxlength 王 "60" 二 结尾 。 

(5) 添加 一 对 单 选 按钮 。 


<p><b> 性 别 < /b> 
< input type= "radio" name= "gender" value=" 男 " /> 男 
< input type= "radio" name= "gender" value= " 女 " /> 女 < /p> 
这 两 个 单 选 按钮 具有 相同 的 名 称 , 这 意味 着 只 能 选中 其 中 的 一 个 。 不 过 ,它们 具有 不 
同 的 值 。 
(6) 添加 一 个 下 拉 菜 单 。 
<p><b> 年 龄 < /b> 
<select name="age"> 
<option value= "0- 29"> 30 岁 以 内 < /option> 
<option value= "30- 60">30 岁 至 60 岁 < /option> 
<option value="60+ "> 60 岁 以 上 < /option> 
</select>< /p> 
select 标签 开始 创建 下 拉 菜 单 ,然后 每 个 option 标签 将 创建 选项 列表 中 的 另 一 行 
(7) 添加 一 个 多 行文 本 框 ,用 于 输入 个 人 详细 信息 。 
<p><b> 个 人 简介 < /b> 
<textarea name= "comments" rows="3" cols="40">< /textarea>< /p> 
textarea 不 同 于 text 输入 框 ; 它 会 展示 为 一 个 方 框 ,而 不 是 一 个 单行 文本 框 。 它 允许 
输入 更 多 的 信息 ,并 且 可 用 于 获取 用 户 注释 。 
(8) 完成 表单 。 
</fieldset> 
<div align= "center"> 
<input type= "submit" name= "submit" value= "提交 " /> 
</div> 
< /form> 
第 一 个 标签 将 会 关闭 在 第 (3) 步 打开 的 fieldset; 然 后 会 创建 一 个 submit 按钮 ,并 使 
用 div 标签 将 其 居中 ;最 后 ,关闭 表单 。 
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(9) 完成 HTML 页 面 。 


< /body> 

</html> 

(10) 将 文件 另存 为 form. html, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 查看 它 ( 参 
见 图 6-1)。 


国人 EECEI EC 
请 输入 您 的 信息 : 
姓名 : 


Email: 


性 别 日 男 日 女 


年 齿 30 岁 以 内 ~ 


个 人 简介 


因为 这 个 页 面 只 包含 HTML, 所 以 使 用 了 . html 扩展 名 ,但 是 ,也 可 以 使 用 . php ,而 
不 会 引起 任何 错误 ,因为 PHP 标签 外 面 的 代码 将 被 视 作 HTML。 


6.1.2 处 理 HTML 表单 


既然 已 经 有 一 个 HTML 表单 (参见 脚本 6-1) ,下 面 将 编写 一 个 基本 的 PHP 框架 脚 
本 来 处 理 它 。 这 个 处 理 表单 的 脚本 ,将 对 其 接收 到 的 数据 执行 某 些 操作 。 在 本 章 中 ,脚本 
将 反复 把 数据 发 送 回 Web 浏览 器 ,但 是 ,在 后 面 的 示例 中 ,还 可 以 将 这 些 数据 存储 在 
MySQL 数据 库 中 等 。 

PHP 的 最 佳 特性 之 一 就 是 能 够 与 HTML 表单 极 好 地 进行 交互 , 它 能 够 非常 方便 地 
处 理 表单 数据 。PHP 脚本 访问 表单 时 不 需要 一 个 解析 例 程 (CGI 脚本 则 需要 ) ,而 是 把 信 
息 存 储 在 特殊 的 超 全 局 变量 中 。PHP 会 自动 填充 两 个 超 全 局 变量 , 即 $_GET 和 $_ 
POST ,它们 分 别 保存 着 来 自 get 或 post 方法 的 数据 。 这 种 特性 让 创建 表单 并 接收 数据 
变 得 特别 简单 。 

例如 ,假设 有 一 个 表单 ,其 中 有 一 个 输入 控件 ,定义 如 下 : 


< input type= "text" name= "weight" size="20" /> 


接收 表单 数据 的 PHP 页 面 将 把 用 户 输入 这 个 表单 元 素 中 的 内 容 赋 予 一 个 特殊 的 超 
全 局 变量 中 ,若是 通过 post 发 送 的 表单 项 会 被 保存 在 $_POST['weight]] 里 ;若是 通过 
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get 发 送 的 表单 项 会 被 保存 在 $_GET['weight"] 里 。 使 拼写 和 大 小 写 完 全 匹配 非常 重要 ， 
因为 PHP 对 变量 名 区 分 大 小 写 。 然 后 可 以 像 任 何其 他 变量 一 样 使 用 $ _POST['weight"] 
或 $_GET['weight]] 这 种 格式 的 超 全 局 变量 : 打印 它 、 在 数学 计算 中 使 用 它 .连接 它 等 。 
1. 轻松 获取 表单 数据 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 ,从 HTML 开始 (参见 脚本 6-2) 。 


脚本 6-2 这 个 脚本 接收 并 打印 出 输入 到 HTML 表单 (参见 脚本 6-1) 中 的 信息 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0rg/TR/xhtml1l/DTD/xhtmll- transitional .dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv= "Content- TYPe”content= "text/html; charset=gb2312" /> 
6 ”<title> 表 单反 馈 信息 < /title> 

7 </head> 

8 <body> 

9 


10 <?php #5Script 6-2-handle form.php 
11 $name=$_POST['name']; 

12 $email=$_POsT['email']; 

13 $gender=$_POsT['gender']; 

14 $age=$_POsT['age']; 

15 $coments=$_POST['comments']} 


17 echo "<p> 谢 谢 您 <b> $name< /b>, 您 的 信息 如 下 :<br />"; 
18 echo "Email:<b>$email</b><br />"; 

19 echo "性别 :<b>$gender< /b><br />"; 

20 echo "年 龄 层 :<b> $age< /b><br />"; 

21 echo "详细 介绍 :<b> $ camments< /b><br /></p>"; 

22 echo "<p> 我 们 将 回复 您 至 <i> $email< /i> .</p>\n"; 


24 ?> 
25 </body> 
26 </html> 


(2) 添加 PHP 开始 标签 ,并 创建 表单 数据 变量 的 简写 版 本 。 


<?php #5Script 6-2-handle form.php 
$name=$_POST['name']; 
$email=$_POsT['email']; 

$ gender=$_POsT['gender']; 
$age=$_POST['age']; 


$ comments=$_POST['comments']; 


依据 以 前 概括 的 规则 ,对 于 输入 到 名 称 表单 输入 框 ( 其 name 值 为 name) 中 的 数据 ， 
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通过 post 方法 传递 ,可 以 通过 变量 $_POST['mname"] 来 访问 它 ; 对 于 输入 到 电子 邮件 表单 
输入 框 (其 name 值 为 email) 中 的 数据 ,可 以 通过 $_ POST [email] 来 访问 它 。 这 同样 适 
用 于 输入 的 其 他 数据 。 此 外 ,这 里 的 变量 的 拼写 和 大 小 写 必 须 与 HTML 表单 中 相应 的 
name 值 完全 匹配 。 

(3) 打印 出 接收 到 的 名 称 、 电 子 邮 件 和 其 他 值 。 


echo "<p> 谢 谢 您 <b> Sname< /b>，, 您 的 信息 如 下 :<br />"; 
echo "Email:<b> $email< /b><br />"; 

echo "性 别 :<b> 5 gender< /b><br />"; 

echo "年 龄 层 :<b> $age< /b><br />"; 

echo "详细 介绍 :<b> $ comments< /b><br />< /p>"; 

echo "<p> 我 们 将 回复 您 至 <i>$ email< /i> .< /p> \n"; 


将 使 用 echo() 语 句 、 双 引号 ,以 及 很 少 一 点 HTML 格式 化 效果 简单 地 打印 出 提交 
的 值 。 

(4) 完成 脚本 ,完成 HTML 页 面 。 

?> 

< /body> 

</html> 

(5) 将 文件 另存 为 handle_form. php, 上 传 到 Web 服务 器 上 与 form. html 相同 的 目 
录 中 ,并 在 Web 浏览 器 中 测试 这 两 个 文档 (参见 图 6-2 和 图 6-3) 。 


请 输入 您 的 信息 ， 
姓名 : Jary 


Email: nary163@126.con 


性 别 目 男 加 女 


年 龄 30 风 以 内 ~ 


Hello World! 


个 人 简介 


图 6-2 要 测试 handle_form. php, 必 须 先 填写 表单 


如 果 在 提交 表单 之 后 看 到 一 个 空白 页 面 , 首 先 可 检查 HTML 源 文件 来 寻找 HTML 
错误 ,然后 确认 PHP 配置 中 的 display_errors 是 打开 的 。 

如 果 PHP 脚本 在 应 该 打印 变量 的 地 方 显示 空白 . 则 意味 着 变量 没有 赋值 。 可 能 的 
原因 是 : 无 法 在 表单 中 输入 值 ; 变 量 名 拼写 错误 或 弄 错 了 它 的 大 小 写 。 
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谢谢 您 Wary， 您 的 信息 如 下 
Email :mary1638126. com 


性 别 : 女 
年 龄 层 ，0-2 
详细 介绍 ，Hello World! 


我 们 将 回复 您 至 mary7638126. com 


图 6-3 脚本 应 该 显示 像 这 样 的 结果 


如 果 看 到 任何 Undefined variable: variablename( 未 定义 的 变量 : 变量 名 ) 错 误 ,这 是 
因 于 引用 的 变量 没有 赋值 ,并 且 PHP 设置 了 最 高 级 的 错误 报告 。 

另外 ,与 这 些 数据 相关 的 还 有 另 一 个 超 全 局 变量 $_REQUEST, 它 包含 来 自 GET 和 
POST 的 全 部 数据 (还 包含 cookie 数据 )。 如 果 不 知道 哪个 方法 在 提供 数据 ,那么 使 用 这 
个 变量 将 是 非常 方便 的 。 在 默认 情况 下 ,在 $_REQUEST 中 POST 数据 会 覆盖 GET 数 
据 。 改 变 初始 配置 中 的 variables_order 可 以 改变 覆盖 次 序 。 

在 一 般 应 用 中 ,最 好 明确 是 要 获得 GET 还 是 POST 数据 ,这 样 可 以 得 到 更 强健 的 代 
码 ,避免 错误 变量 值 带 来 意外 的 结果 。 

当 处 理 源 于 表单 的 字符 串 时 ,使 用 trim() 函 数 也 是 一 个 好 主意 , 它 会 从 值 的 两 端 删 
除 多 余 的 空白 。 例 如 , $ name 一 trim($name) 。 

2. 获取 表单 数据 的 多 维 数 组 

在 脚本 6-1 中 使 用 的 表单 控件 包括 单行 文本 框 . 单 选 按钮 .下 拉 菜 单 及 多 行文 本 框 ， 
这 些 控件 都 具有 唯一 的 值 。 在 表 6-1 所 介绍 的 常用 表单 控件 中 ,除了 复 选 框 之 外 ,其 他 控 
件 都 只 具有 一 个 值 。 下 面 就 来 详细 介绍 如 何 获取 表单 中 复 选 框 中 选中 的 多 个 值 。 

(1) 在 文本 编辑 器 中 修改 脚本 6-1( 参 见 脚本 6-3) 。 


脚本 6-3 在 简单 的 HTML 表单 中 添加 复 选 框 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 


1 
2 
3 
4 
5 ”<meta http- equiv= "Content- Type" content="text/html; charset=gb2312" /> 
6 <title> 简 易 的 表单 < /title> 

7 </head> 

8 <body><!-- Script 6-3-about.html--> 

9 

10 <formaction= "handle about.php" method="post"> 

11 <fieldset><legend> 请 输入 您 的 信息 :< /legend> 

12 <p><b> 姓 名 :< /b> 
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13 <input type= "text" name= "name"” size= "20" maxlength= "40" />< /p> 
14 <p><b> Email:< /b> 

15 <input type= "text" name= "email" size= "40" maxlength= "60" />< /p> 
16 <p><b> 性 别 < /b> 

17 <input type= "radio" name= "gender" value=" 男 " /> 男 

18 <input type= "radio" name= "gender" value=" 女 " /> 女 < /p> 

19 <p><b> 年 龄 < /b> 

20 < select name= "age"> 

21 <option value= "0- 29"> 30 岁 以 内 < /option> 

22 <option value= "30- 60"> 30 岁 至 60 岁 < /option> 

23 <option value= "60+">60 岁 以 上 < /option> 

24 </select>< /p> 

25 <p><b> 爱 好 :< /b> 

26 <input type= "checkbox" name= "interests[]" value= "音乐 " /> 音乐 
27 <input type= "checkbox" name= "interests[]" value= "电影 " /> 电影 
28 <input type= "checkbox" name= "interests[]" value= "看 书 " /> 看 书 
29 <input type= "checkbox" name= "interests[]" value= "运动 " /> 运动 
30 < input type= "checkbox" name= "interests[]" value= "其 他 " /> 其 他 < /p> 
31 <p><b> 个 人 简介 < /b> 

32 <textarea name= "comments" rows="3" cols="40">< /textarea>< /p> 
33 </fieldset> 

34 <div align= "center"> 

35 <input type="submit" name= "submit" value= "提交 " /> 

36 </div> 

37 </form> 

38 </body> 

39 </html> 


(2) 在 表单 原 有 基础 上 ,创建 一 系列 复 选 框 ,使 用 户 可 以 选择 他 们 的 兴趣 。 


<p><b> 爱 好 :< /b> 

< input type= "checkbox" name= "interests[]" value= "音乐 " /> 音乐 

< input type= "checkbox" name= "interests[]" value= "电影 " /> 电影 

< input type= "checkbox" name= "interests[]" value= "看 书 " /> 看 书 

< input type= "checkbox" name= "interests[]" value= "运动 " /> 运动 

<input type= "checkbox" name= "interests[]" value= "其 他 " /> 其 他 < /p> 

表单 的 “爱好 ?部 分 可 供用 户 选择 多 个 选项 。 在 这 种 情况 下 ,可 以 为 多 个 不 同 的 输入 
使 用 两 个 命名 模式 ,为 它们 提供 与 其 值 匹配 的 名 称 ( 音 乐 .电影 等 ) ,或 者 使 用 一 个 数组 。 
在 这 里 使 用 interests[ ] 作 为 名 称 , 在 PHP 处 理 表单 复 选 框 时 , 它 将 在 $_POST['interests] 中 
创建 一 个 数组 ,用 于 存储 用 户 选 中 的 所 有 复 选 框 。 

(3) 将 文件 另存 为 about. html。 

现在 将 创建 handle_about. php ,来 使 用 表单 提交 的 $_POST[Yinterests] 多 维 数组 。 
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(4) 在 文本 编辑 器 中 修改 脚本 6-2( 参 见 脚本 6-4) 。 


脚本 6-4 这 个 脚本 将 打印 出 $_POST['interests]] 的 值 , 它 是 多 维 $_POST 数组 的 一 
部 分 。 


1 < !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN” 

2 "http://www.w3.0org/TR/xhtml1/DTD/xhtmll1- transitional.dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv="Content- Type" content="text/html; charset=gb2312" /> 
6 ”<title> 表 单反 馈 信息 < /title> 

7 </head> 

8 <body> 

多 


10 <?php #5Script 6- 4-handle about.php 
11 $name=$_POST['name']; 

12 $email=$_ POST['email']; 

13 $gender=$_POsT['gender']; 

14 $age=$_POsT['age']; 

15 $interests=$_POST['interests']; 


16 $coments=$_POST['comments']; 


18 echo "<p> 谢 谢 您 <b> Sname< /b>，, 您 的 信息 如 下 :<br />"; 
19 echo "Email:<b>$ email< /b><br />"; 

20 ”echo "性 别 :<b>$ gender< /b><br />"; 

21 echo "年 龄 层 :<b> $age< /b><br />"; 

22 echo "您 的 爱好 包括 :<ul>"; 

23 foreach ($ interests as $value) 

24 echo "<1li>" . $value . "</li>"; 

25 echo "</ul>"; 

26 ”echo "详细 介绍 :<b> $ comments< /b><br /></p>"; 

27 echo "<p> 我 们 将 回复 您 至 <i>$ email< /i> .</p>\n"; 


29 3> 
30 </body> 
31 </htm> 


(5) 创建 表单 数据 变量 的 简写 版 本 ,将 用 户 的 爱好 选项 创建 为 $interest 数组 。 
$ interests=$_POST['interests']; 


(6) 添加 处 理 多 维 数组 的 代码 ,打印 提交 的 爱好 。 这 里 把 爱好 显示 为 HTML 列表 ， 
所 以 先 在 这 里 打印 初始 的 无 序列 表 标 签 ( 二 ul 二 )。 


echo "您 的 爱好 包括 :<ul>"; 
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(7) 打印 出 所 有 选择 的 兴趣 ,并 关闭 HTML 列表 。 


foreach ($ interests as $value) 
echo "<1i>" .$value . "</li>"; 

echo "</ul>"; 

这 个 循环 将 遵循 数组 遍历 的 foreach 语法 ,来 访问 $ _POST[L'interests 的 每 个 元 素 。 
由 于 表单 复 选 框 本 身 的 性 质 , 只 有 那些 被 选中 的 复 选 框 才 具有 $_POST[L'interests"] 数 组 
中 的 值 。 在 循环 自身 内 ,将 会 在 HTML 列表 标签 内 打印 每 个 值 。 

(8) 将 文件 另存 为 handle_about. php ,与 about. html 一 起 上 传 到 Web 服务 器 ,并 在 
Web 浏览 器 中 测试 它们 (参见 图 6-4 和 图 6-5)。 


请 输入 您 的 信息 ， 
姓名 Jary 


Email: nary163@126.con 
性 别 ， 四 男 加 女 
年 龄 。 50 风 g 内 “~ 
爱好 ， 国 音乐 国电 影 固 看 书 加 运动 回 其 他 


Hello Worldl| 


| 个 人 乔 介 ， 


谢谢 您 Eary， 您 的 信息 如 下 : 
Email :mary1639126- com 


| 详细 介绍 :Hello World! 


| 我 们 将 回复 您 至 mary7638136. com 


图 6-5 所 选 的 爱好 将 作为 HTML 列表 显示 给 用 户 


即使 用 户 只 选择 了 一 个 兴趣 , $_POST['interests] 也 仍然 是 一 个 数组 ,因为 相应 输 
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入 的 HTML 名 称 是 interests[], 方 括号 代表 数组 。 
另外 ,也 可 以 创建 一 个 多 维 数组 , 它 具 有 一 个 HTML 表单 的 选项 菜单 ,允许 进行 多 
重 选择 : 
<select name="interests[]" multiple= "multiple"> 
<option value= "音乐 "> 音乐 </option> 
<option value= "电影 "> 电 影 < /option> 
<option value= "看 书 "> 看 书 < /option> 
<option value= "运动 "> 运动 < /option> 
<option value= "其 他 "> 其 他 < /option> 
</select> 


6.2 表单 验证 


使 用 服务 器 端 数据 验证 ,就 是 利用 PHP 脚本 来 处 理 表 单数 据 。 与 客户 端 验证 相 比 ， 
使 用 服务 器 验证 的 优点 在 于 : 它 更 安全 ,与 所 有 浏览 器 无 颖 对接 ;缺点 是 代价 稍 高 ,用 户 
反馈 慢 、 增 加 了 服务 器 负荷 。 

使 用 服务 器 端 验 证 另 一 大 的 优势 是 : 可 以 用 PHP 对 校 验 规则 进行 任意 的 修改 ,利用 
PHP 的 多 种 函数 和 灵活 特点 ,可 以 很 方便 地 更 改 校 验 的 数据 类 型 ,长度 ,以 及 检查 文本 框 
范围 内 的 号 码 等 。 

另外 ,如 果 用 PHP 连接 MySQL 数据 库 才 能 验证 用 户 名 或 密码 的 正确 性 ,在 这 种 情 
况 下 ,根本 不 可 能 使 用 客户 端 脚 本 。 


6.2.1 验证 表单 是 否 提交 


6. 1 节 关 于 使 用 PHP 处 理 表单 的 示例 都 使 用 了 两 个 单独 的 文件 : 一 个 用 于 显示 表 
单 , 另 一 个 用 于 接收 表单 。 虽 然 这 种 方法 没有 任何 错误 ,但 是 把 整个 过 程 集中 到 一 个 脚本 
中 更 有 利 。 
为 了 让 一 个 页 面 同时 显示 和 处 理 表单 ,必须 使 用 一 个 条 件 语句 检查 应 该 采取 哪 种 
动作 : 
if(/* 表单 已 被 提交 * /) { 
// 处 理 表单 
} else { 
// 显 示 表 单 
} 


为 了 确定 表单 是 否 已 提交 ,可 检查 $ _POST 变量 是 否 已 设置 ,当然 ,假定 表单 使 用 
POST 方法 。 例 如 ,可 以 为 提交 按钮 创建 名 称 .因为 只 有 按 下 提交 按钮 ,整个 表单 才 会 被 
提交 : 


<input name= "submitted" type= "submit" value= "提交 " /> 
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然后 编写 用 于 表单 提交 的 条 件 测试 代码 : 


if(isset($_ POST['submitted'])){ 
// 处 理 表单 

} else { 
// 显 示 表 单 

} 


如 果 按 下 提交 按钮 提交 了 表单 ,$_POST[submitted] 会 具有 一 个 值 ,因此 会 处 理 表 
单 。 使 用 上 面 的 代码 ,如果 提 交 了 表单 ,脚本 将 会 处 理 它们 ,并 会 在 每 次 加 载 页 面 时 显示 
该 表单 。 


6.2.2 验证 表单 数据 


与 处 理 HTML 表单 相关 的 一 个 关键 概念 是 验证 表单 数据 。 从 错误 管理 和 安全 性 两 
方面 考虑 ,作为 开发 者 或 系统 管理 员 ,永远 都 不 应 该 信任 输入 到 HTML 表单 中 的 数据 。 
对 于 错误 的 数据 ,无 论 它 是 蓄意 产生 破坏 ,还 是 仅仅 无 意 地 造成 错误 都 需要 Web 架构 师 
针对 预期 的 要 求 对 其 进行 测试 。 

表单 验证 的 第 一 个 目标 是 确保 在 表单 元 素 中 输入 或 选择 了 某 些 内 容 。 第 二 个 目标 是 
确保 提交 的 数据 具有 正确 的 类 型 (数字 ,字符 串 等 ) .正确 的 格式 (如 电子 邮件 地 址 ) 或 特定 
的 可 接受 值 (如 $ gender 应 该 等 于 男 或 女 ) 。 

验证 表单 数据 需要 使 用 条 件 语 句 以 及 许多 函数 .运算 符 和 表达 式 。 常 用 的 函数 有 
两 个 : 

。 isset() 函 数 , 它 用 于 测试 一 个 变量 是 否 具有 值 ,包括 0.FALSE, 或 者 一 个 空 字符 

串 ,但 不 能 是 NULL。 代 码 类 似 : 


if (isset ($var)){ 

//$var 存在 且 有 值 
Jelse{ 

//$var 不 存在 或 不 具有 值 
} 


。 empty() 函 数 . 它 将 检查 一 个 变量 是 否 具有 空 (empty) 值 ,包括 空 字符 串 .0 .NULL 
或 FALSE。 


证 (empty($var)){ 
/VSvar 为 空 
Jelse{ 
//Svar 不 为 空 
} 


使 用 isset() 函数 的 一 个 问题 是 : 空 字 符 串 测试 为 TRUE, 这 意味 着 它 不 是 验证 
HTML 表单 中 的 文本 输入 和 文本 框 的 有 效 方式 。 要 检查 用 户 输入 到 文本 元 素 ( 如 名 称 、 
电子 邮件 和 注释 ) 中 的 内 容 , 可 以 使 用 empty() 函 数 。 
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下 面 来 演示 如 何 让 同一 个 页 面 显示 和 处 理 表 单 , 以 及 如 何 验证 表单 元 素 的 值 , 让 我 们 
创建 一 个 简单 的 注册 页 面 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 ,从 HTML 开始 (参见 脚本 6-5) 。 


脚本 6-5 这 个 脚本 将 会 显示 一 个 简单 的 表单 并 处 理 表单 数据 ,检查 必 填 选项 是 否 


为 空 。 


< !DOCTYPE html PUBLIC "- //W3C//DITD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.0rg/1999/xhtml"> 
<head> 
<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title> 注 册 表单 < /title> 
</head> 
<body> 
<p align= "center"> 用 户 注册 <br /> * 号 标志 必 填 < /p> 
< form name= "forml" method= "post" action=""> 
<table width= "500"> 
<tr> 
<td> 用 户 名 * < /td> 
<td>< input name= "name" type= "text" />< /td> 
</tr> 
Ey 
<td> 密 码 *</td> 
<td>< input name= "passl" type= "password" />< /td> 
</tr> 
<tr> 
<td> 确 认 密 码 * < /td> 
<td>< input name= "pass2" type= "password" />< /td> 
</tr> 
<tr> 
<td> Email 地 址 * < /td> 
<td><input name= "email" type= "text" />< /td> 
</tr> 
<tr> 
<td> 性 别 < /td> 
<td>< input type= "radio" name= "gender" value= "1" /> 男 
<input type="radio" name= "gender" value="2" /> 女 
< input name= "gender" type= "radio" value="3" checked= "checked" /> 保密 < /td> 
</tr> 
<tr> 
<td> 出 生日 期 < /ta> 
<td>< input name= "birthday" type= "text" /> // 格 式 1900- 01- 01< /td> 
</tr> 
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< 
<td> 爱 好 < /td> 
<td>< input name= "interests[]" type= "checkbox" value= "电影 " checked= 
"checked" /> 电影 
< input name= "interests[]" type= "checkbox" value= "音乐 " /> 音乐 
< input name= "interests[]" type= "checkbox" value= "运动 " /> 运动 
<input name= "interests[]" type= "checkbox" value= "看 书 " /> 看 书 
<input name= "interests[]" type= "checkbox" value= "其 他 " /> 其 他 < /td> 
</tr> 
<tr> 
<td> 主 页 < /td> 
<td>< input name= "web" type= "text" value= "http://"” /></td> 
</tr> 
Sr 
<td> 个 人 简介 < /td> 
<td><textarea name= "more" cols="30" rows= "5">< /textarea> 
</td> 
</tr> 
<tr align= "center" valign= "middle"> 
<td colspan= "2"><input type= "submit" name= "Submit" value= "提交 " /> 
<input type="reset" name= "reset" value=" 重 置 " />< /td> 
</tr> 
</table> 


< /form> 


< ?php #Script 6- 5- regist.php 


if($_POST['Submit']){ 

if(empty($_FOST["name'"])){ 
echo "用 户 名 不 能 为 空 ."; 
exit; 

} 

if (empty($_POST['pass1"'])){ 
echo "密码 不 能 为 空 。"; 
exit; 

} 

if (empty($_POST['pass2"'])){ 
echo "确认 密码 不 能 为 空 ." 
exit; 

. 

if(empty($_FOST["email'])){ 
echo "Email 不 能 为 空 . "7 
exit; 

} 

if($_POST['pass1'] !==$_POST['pass2"']){ 
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81 echo "密码 与 确认 密码 不 同 ,请 重新 输入 。"; 
82 exit; 

83 } 

84 ?> 


85 <p align= "center"> 您 的 输入 为 :< /p> 
86 <table width="500"> 


87 <tr> 
88 <td> 用 户 名 </td><td><?Php echo $_POsT['name'];?>< /td> 
89 </tr> 

90 <tr> 

91 <td> 密 码 < /td><td><?php echo $_POsST['pass1'];?>< /td> 
92 </tr> 

93 <tr> 

94 <td> Email 地 址 < /td><td><?php echo $_POsT['email'];?></td> 
95 </tr> 

96 <tr> 

97 <td> 性 别 < /td> 

98 <td> 

99 <?php 

100 switch($_POsT['gender']) { 

101 case 1: 

102 echo " 男 "; 

103 break; 

104 case 2: 

105 echo " 女 "y 

106 break; 

107 Case 3: 

108 echo "保密 "; 

109 break; 

110 } 

111 ?> 

112 </td> 

113 </tr> 

114 <tr> 

115 <td> 出 生日 期 </td><td><?Php echo $_POST['birthday'];?></td> 
116 </tr> 

117 <tr> 

118 <td> 爱 好 < /td> 

119 <td> 

120 <?php 

121 foreach ($ interests as $value) { 

122 echo $value; 

223 echo "gnbsp; gnbsp;"; 


124 } 
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125 ?> 
126 </td> 
127 </tr> 
128 <tr> 


129 <td> 主 页 < /td> 
130 <to> <a href=<?php echo $_FOST["web'];?>><?Fhp echo $_POST['web'];? > 
</a></td> 


131 </tr> 
132 <tr> 
133 <td> 个 人 简介 < /td><td><?php echo $_POST['more'] ?>< /td> 
134 </tr> 
135 </table> 
136 

137 <?php 
138 } 

139 ?> 

140 </body> 
141 </htm> 


对 于 这 个 示例 ,为 了 阅读 方便 ,省 略 了 部 分 用 于 排版 和 美化 的 HTML 和 CSS 代码 。 
(2) 显示 HTML 表单 (代码 部 分 为 从 第 10 行 至 第 60 行 ) ,此 表单 涉及 了 大 部 分 的 表 
单 控件 。 


<p align= "center"> 用 户 注册 <br /> * 号 标志 必 填 < /p> 
< form name= "forml" method= "post" action=""> 


表单 自身 相当 容易 理解 ,其 中 只 包含 两 个 新 技巧 : 第 一 ,action 属性 为 空 , 使 得 表单 
提交 到 这 个 页 面 ,而 不 是 提交 到 另 一 个 页 面 ;第 二 ,有 一 个 称 为 Submit 的 提交 按钮 ,其 值 
为 1。 这 也 可 以 作为 一 个 标志 变量 .通过 检查 其 存在 与 否 来 确定 是 否 处 理 表 单 ( 见 第 
(3) 步 中 或 第 63 行 上 的 主 条 件 )。 由 于 只 使 用 isset 检测 它 是 否 具有 值 ,所 以 可 以 赋予 它 
任何 值 。 

(3) 编写 处 理 表单 的 条 件 语句 。 

if(isset ($_POST['Submit"])){ 

如 前 所 述 , 通 过 检查 是 否 设置 了 某 个 表单 元 素 ( 如 $_POST['Submit]]) , 即 可 测试 是 
否 提 交 了 表单 。 该 变量 将 被 关联 到 表单 中 提交 按钮 。 

(4) 验证 表单 。 

if (empty($_POsT['name'])){ 


echo "用 户 名 不 能 为 空 。"; 


exit; 
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证 (empty($_POST["passl"])){ 
echo "密码 不 能 为 空 。"; 
exit; 
} 
if (empty($_ POsT['pass2'])){ 
echo "确认 密码 不 能 为 空 。"; 
exit; 
} 
if(empty($_POST['email'])){ 
echo "Email 不 能 为 空 。"; 
exit; 
} 
if($_POsT['pass1']!==$_POsT['pass2°']){ 
echo "密码 与 确认 密码 不 同 ,请 重新 输入 。"; 
exit; 
} 
这 里 的 验证 非常 简单 : 它 只 会 检查 4 个 提交 的 变量 是 否 为 空 ,以 及 两 次 密码 的 值 是 
否 相 等 。 若 有 任意 一 项 为 空 ,将 提示 错误 信息 ,并 要 求 用 户 重 填 。 每 个 if 语句 中 都 有 一 
名 “exit”,PHP 中 的 exit 代码 将 结束 当前 PHP 代码 块 的 执行 。 
(5) 如 果 验 证 通过 了 所 有 的 测试 ,将 会 以 表格 格式 显示 用 户 输 入 的 信息 (代码 部 分 为 
从 第 85 行 至 第 135 行 ) 。 


<p align= "center"> 您 的 输入 为 :</p> 
<table width= "500"> 


</tr> 
</table> 


其 中 使 用 文本 输入 框 提交 的 表单 数据 ,例如 ,用 户 名 、 密 码 .Email、 生 日 ,主页 及 个 人 简介 ， 
这 些 数 据 的 输出 可 以 直接 使 用 echo( ) 方 法 ,输出 对 应 的 $_POST 变量 。 性 别 的 输出 使 用 
了 switch 结构 ,因为 给 用 户 看 到 的 内 容 是 中 文 的 ,而 实际 存储 到 数据 库 的 值 应 该 越 简洁 
越 好 。 


< ?Php 
Switch ($_POST["gender"]){ 

case 1: 
echo " 男 "; 
break; 

case 2: 
echo " 女 "; 
break; 

case 3: 


echo "保密 "; 
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break; 


复 选 框 爱好 的 输出 ,使 用 数组 的 foreach 循环 ,并 将 每 个 输出 之 间 用 两 个 空格 字符 
隔 开 : 
<?php 
foreach ($ interests as $value){ 
echo $value; 
echo "gnbsp; gnbsp;"; 


2 

个 人 主页 的 输出 ,不 但 显示 主页 内 容 , 还 为 其 制作 了 超 链接 。 

<a href=<?php echo $_POST['web'];?>><?php echo $_POST['web'];?></a> 
(6) 完成 条 件 语句 并 关闭 PHP 标签 。 


<?php 
} 
?> 
这 里 完成 了 验证 条 件 语 句 (第 (3) 步 ), 如 果 4 个 提交 的 值 只 要 有 一 个 为 空 ,就 会 打印 
一 个 错误 。 最 后 的 结束 花 括号 会 关闭 isset($ _POST['Submit"]) 条 件 语 句 。 最 后 ,会 关 
闭 PHP 部 分 ,因此 无 须 使 用 echo() 即 可 创建 表单 。 
(7) 完成 页 面 。 
< /body> 
</html> 
(8) 将 文件 另存 为 regist. php, 存 放 在 Web 目录 中 ,并 在 Web 浏览 器 中 测试 它 ( 参 见 
图 6-6、 图 6-7 和 图 6-8) 。 
一 个 经 过 深思 熟 虑 的 良好 表单 可 以 让 用 户 在 填写 表单 时 知道 哪些 字段 是 必须 填写 
的 ,并 且 在 适当 的 地 方 指出 了 那个 字段 的 格式 (如 日 期 或 电话 号 码 ) 。 
验证 文本 输入 的 另 一 种 方式 是 : 使 用 strlen() 函数 来 查看 是 否 输入 了 0 个 以 上 的 
if(strlen($var)>0){ 
//Svar 不 为 空 
Jelse{ 
//$var 为 空 
} 
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用 户 注册 
* 号 标志 必 填 


男 © 女 国保 密 
/格式 1900-01-01 

团 电影 固 音乐 回 运 动 回 看 书 固 其 他 
htm 


© 男 © 女 时 保密 
/4 格式 1900-01-01 


团 电影 ” 固 音乐 ” 固 运动 ” 固 看 书 固 其 他 
http:// 


6-7 ”如 果 必 填 项 为 空 , 就 会 显示 一 条 出 错 消 息 
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[= /5 mE | 
| E ntpy/localhosybe P - © x|| Em x Pox 
用 户 注册 二 
* 号 标志 必 填 
用 户 名 * 
宣 到 * 
确认 密码 
Emaitig 址 * 
性 别 © 男 © 女 全 保密 
出 生日 期 CC |] /1 格式 1900-01-01 
要 好 团 电 影 回音 乐 回去 动 固 看 书 固 其 他 
主页 http:// 
个 人 简介 
[EC 
您 的 输入 为 ， 
人 
密码 
Email 地 址 mary163@126.com 
性 别 皮 
出 生日 期 hi988-11-11 
要 好 由 影 音乐 运动 
主页 Mhttp:/ /www.maryblog.com 


个 人 简介 MHeho Worid! 


图 6-8 ”该 页 面 执行 结果 ,重新 显示 表单 内 容 


6.2.3 避免 表单 多 次 提交 


出 于 一 些 原因 ,我 们 需要 避免 表单 被 多 次 重复 提交 ,否则 就 会 得 到 重复 的 记录 。 这 可 
能 是 由 于 粗心 的 用 户 在 提交 表单 之 后 又 单 击 了 
“刷新 ”按钮 ,使 数据 再 次 提交 ,如 图 6-9 所 示 。 更 
坏 的 情况 是 有 人 故意 通过 反复 提交 同一 表单 来 攻 A Sean web ws 
重新 发 送 您 以 前 提交 的 信息 . 
击 站 点 。 这 样 不 仅 会 造成 数据 混乱 问题 ,还 会 知 
食 宝 贵 的 站 点 资源 。 应 单 去 “取消 适 旬 重复 交易 . 
可 以 从 客户 端 和 服务 器 端 同时 着 手 , 设 法 避 


免 同 一 表单 的 重复 提交 。 | =— 
1. 使 用 客户 端 脚本 


提 到 客户 端 脚本 ,经 常 使 用 的 是 JavaScript 图 6-9 刷新 会 引起 重复 提交 表单 
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进行 常规 输入 验证 。 在 下 面 的 例子 中 ,我 们 使 用 它 处 理 表单 的 重复 提交 问题 ,请 看 下 面 的 
代码 : 


< form method= "post" name= "register" action= "test.Php" enctype= "multipart/fom- data"> 
<input name= "text" type= "text" id= "text" /> 
<input name= "submit" value= "提交 " type= "button" onClick= "document.register . submit. 
value= ' 正 在 提交 ,请 等 待 ... '; document. register. submit. disabled= true; document. 
Tegister.submit ();" /> 

< /form> 


当 用 户 单 击 “ 提 交 ” 按 钮 后 ,该 按钮 将 变 为 灰色 不 可 用 状态 ,如 图 6-10 所 示 。 


Email: 


性 别 日 男 日 女 
年 龄 50 风 愉 内 “~ 


个 人 简介 


正在 反共 , 清苦 待 ,- 


6-10 利用 客户 端 脚本 防止 重复 提交 表单 


上 面 的 例子 中 使 用 OnClick 事件 检测 用 户 的 提交 状态 ,如 果 单 击 了 “提交 ”按钮 ,该 按 
钮 立即 置 为 失效 状态 ,用 户 不 能 单 击 按钮 再 次 提交 。 

还 有 一 个 方法 ,也 是 利用 JavaScript 的 功能 ,但 是 使 用 的 是 OnSubmit() 方 法 ,如 果 已 
经 提交 过 一 次 表单 ,将 立即 弹出 对 话 框 ,代码 如 下 : 


< script language= "javascript"> 
Var submitcount=07 
function submitOnce (form) { 
if(submitcount==0){ 
submitcount++? 
return true; 
} elsef 
alert ("正在 操作 ,请 不 要 重复 提交 ,谢谢 !") ; 
return false; 
} 
} 
</script> 
< form name= "the fom" method= "post" action="" onSubmit= "return submitOnce (this)"> 
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< input name= "text" type= "text" id= "text" /> 

<input name= "cont" value= "提交 " type= "submit" /> 

< /form> 

在 上 例 中 ,如 果 用 户 已 经 单 击 “ 提 交 ” 按 钮 , 则 该 脚本 会 自动 记录 当前 的 状态 ,并 将 
submitcount 变量 自 加 1; 当 用 户 试图 再 次 提交 时 ,脚本 判断 submitcount 变量 值 非 零 , 提 
示 用 户 已 经 提交 ,从 而 避免 重复 提交 表单 。 

2. 使 用 header 函数 转向 

除了 上 面 的 方法 之 外 ,还 有 一 个 更 简单 的 方法 , 那 就 是 当 用 户 提交 表单 ,服务 器 端 处 
理 后 立即 转向 其 他 的 页 面 。 

PHP 的 内 置 函 数 header() 函 数 可 用 于 将 浏览 器 从 当前 页 面 重 定 向 到 另 一 个 页 面 。 

要 用 header() 重 定向 Web 浏览 器 ,可 使 用 代码 : 

header ('location:submits_success.php'); 

由 于 这 应 该 是 在 当前 页 面 上 发 生 的 最 后 一 件 事 , 因 为 浏览 器 很 快 会 离开 它 , 这 一 行 之 
后 通常 紧 接 着 exit() 函 数 ,以 阻止 当前 脚本 的 执行 。 

这 样 ,即使 用 户 使 用 刷新 键 ,也 不 会 导致 表单 的 重复 提交 ,因为 已 经 转向 新 的 页 面 ,而 
这 个 页 面 脚本 已 经 不 理会 任何 提交 的 数据 了 。 

header() 函 数 的 应 用 更 常用 于 PHP 与 数据 库 的 交互 。 

关于 header() 函 数 要 记 住 的 绝对 关键 点 是 : 必须 把 任何 内 容 发 送 给 Web 浏览 器 之 
前 调用 它 。 这 包括 HTML 代码 ,甚至 是 空白 行 。 如 果 代码 在 调用 header() 之 前 具有 任 
何 的 echo() 或 print() 语 句 ,把 空白 行 保 持 在 PHP 标签 外 面 ,或 者 包含 做 任何 事情 的 文 
件 , 则 会 看 到 出 错 信息 ,如 图 6-11 所 示 。 


TT ox 


(SO le | 


Warning: Cannot modify header information - headers already 
sent by (output started at C:VAppServ\wwwNphpbookNch06 


\Untitled-1. php:2) in C:\AppServ\www\phpbook\ch06 
\Untitled-1.php on line 3 


图 6-11 使 用 header() 函 数 时 ,经 常会 看 到 headers already sent 错误 


一 个 避免 出 现 这 个 错误 的 方法 ,可 以 通过 一 些 输 出 缓冲 函数 来 解决 这 个 问题 。 代 
价 是 把 所 有 向 浏览 器 的 输出 都 缓存 在 服务 器 ,直到 下 命令 发 送 它们 。 可 以 在 代码 中 使 用 
ob_start() 及 ob_end_flush() 函 数 来 实现 这 样 的 功能 .或 者 通过 修改 php. ini 中 的 output 
_buffering 配置 选项 来 实现 ,也 可 以 通过 修改 服务 器 配置 文件 来 实现 。 
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6.3 项 目 训 练 


用 户 注 册 功 能 设计 


6.3.1 项 目 说 明 


用 户 表单 是 允许 用 户 自 行 填 写 的 交互 控件 ,因此 安全 性 尤为 重要 ,作为 系统 管理 员 或 
Web 设计 师 , 需 要 对 用 户 提交 的 信息 反复 检查 ,检查 有 效 性 、 正 确 性 、 合 法 性 等 。 

在 检查 数据 正确 性 时 ,分 解 为 两 个 步骤 将 会 带 来 很 大 便利 : 一 是 将 输入 数据 标准 化 ， 
二 是 检查 标准 化 之 后 的 数据 。 输 入 数据 的 “标准 化 ”就 是 将 得 到 的 数据 调整 为 统一 或 是 标 
准 格式 。 各 种 信息 本 来 都 会 有 一 些 不 同 的 表达 方式 ,但 对 于 特定 的 应 用 程序 来 说 ,它们 都 
需要 被 转化 为 某 种 标准 格式 。 之 后 ,就 可 以 对 标准 格式 的 数据 应 用 有 效 性 规则 ,从 而 确定 
输入 的 数据 是 可 用 的 。 

将 数据 检验 分 为 两 个 步骤 可 以 使 过 程 更 加 简单 和 灵活 。 标 准 化 过 程 只 关心 格式 问 
题 ,程序 可 以 让 用 户 以 更 自由 的 方式 提供 信息 ,从 而 具有 更 高 的 用 户 友好 度 。 当 输入 信息 
被 标准 化 之 后 ,检查 过 程 就 简单 多 了 ,因为 信息 格式 是 已 知 的 。 

这 里 我 们 讨论 如 何 使 程序 运行 更 安全 ,这 是 开发 者 的 职责 ,应 该 切实 把 握 ,避免 错误 
发 生 。 因 此 ,需要 开发 一 些 技术 去 避免 因 用 户 不 规范 输入 而 造成 麻烦 。 

一 个 非常 重要 的 技术 是 如 何 从 用 户 的 输入 中 有 效 验 证 数据 ,从 而 保护 网 站 程序 与 数 
据 。 除 了 在 php. ini 文件 中 把 register_globals 设置 为 Off 外 ,还 要 把 错误 级 别 修改 为 E_ 
ALL | E_STRICT, 这 样 就 可 以 阻止 从 外 部 请 求 的 数据 中 生成 全 局 变量 ,后 面 的 设置 是 把 
错误 级 别 设置 为 打开 初始 化 变量 的 警告 错误 。 


6.3.2 设计 思路 


对 于 不 同类 型 数据 的 表单 提交 ,可 以 使 用 不 同 的 方法 来 处 理 。 

1. 验证 电子 邮件 地 址 

电子 邮件 地 址 遵循 特定 的 格式 , 它 必须 由 字符 ,数字 或 下 划 线 开始 ,之 后 可 以 有 任意 
数量 的 上 述 字符 及 句点 ,接着 是 "@? 和 域名 。 域 名 包含 任意 数量 的 “单词 ?及 分 隔 它们 的 
句点 ,而 这 些 “ 单 词 ? 是 由 字符 .数字 和 破 折 号 组 成 的 。 最 后 一 个 后 组 必须 是 2 一 6 个 字符 
长 。 有 些 程序 只 允许 2 或 3 个 字符 ,这 是 不 对 的 ,因为 没有 考虑 到 最 新 的 顶级 域名 ,如 
“InDUSeum。 

可 以 使 用 ereg() 函 数 来 验证 电子 邮件 地 址 ,这 个 函数 专门 用 于 匹配 字符 串 内 的 模式 ， 
而 所 谓 的 模式 就 是 正则 表达 式 。 正 则 表达 式 是 一 种 极其 强大 的 工具 ,在 今天 大 多 数 的 编 
程 语言 中 都 可 以 使 用 它 , 这 是 一 种 精心 设计 的 匹配 模式 系统 。 关 于 正则 表达 式 的 具体 用 
法 可 参考 相关 帮助 。 

以 下 是 验证 电子 邮件 地 址 的 代码 范例 : 


ereg("“^([a-zA-20-9 -1])+@ ([a- zA-20-9 —-])+(\.[a-zA-20-9 —]{2,6})+",$address); 
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当然 ,这 段 代 码 只 是 检验 输入 的 电子 邮件 地 址 是 否 属于 有 效 范 围 ,而 且 只 允许 使 用 最 
常见 的 形式 。 现 实 中 存在 着 一 些 其 他 形式 ,需要 对 代码 进行 少许 修改 才能 适应 。 特 别 是 
主机 名 应 该 可 以 用 包围 在 方 括号 里 的 IP 地 址 表示 ;用 户 名 也 应 该 可 以 用 双 引 号 包围 的 字 
符 串 表示 ,其 中 允许 包含 空格 。 但 在 当今 的 互联 网 上 ,这 种 情况 是 很 少见 的 。 

2. 验证 日 期 

PHP 具有 非常 出 色 的 日 期 工具 , 极 大 地 简化 了 日 期 标准 化 和 检验 工作 。 特 别 是 
strtotime( ) 函数 能 够 把 绝 大 多 数 表示 日 期 的 字符 串 转化 为 UNIX 时 间 戳 ,如 果 转 化 失败 ， 
它 会 返回 False。 这 个 工具 拥有 非常 强大 的 功能 ,几乎 不 需要 我 们 再 做 其 他 工作 。 在 转化 
为 时 间 惟 之 后 ,我 们 就 可 以 根据 需要 使 用 date() 将 它 格式 化 为 任何 式样 。 

以 下 为 实现 日 期 标准 化 和 检验 的 代码 范例 : 


$birthday= "1988-11-11';» 
$output= strtotime ($birthday); 
if ($output===false){ 
echo "日 期 格式 不 正确 "; 
Jelse{ 
echo date('Y 年 m 月 d 日 ',$ output); 
} 


需要 提醒 的 是 ,strtotime() 函数 的 某 些 功能 在 一 开始 可 能 会 让 人 感到 迷惑 。 例 如 它 
允许 任何 一 个 月 的 日 期 都 可 以 到 31 日 ,但 如 果 相应 的 月 份 不 应 该 有 31 天 , 它 会 自动 变化 
到 有 效 的 日 期 ,因此 11/31/2005 会 转化 为 12/1/2005。 它 还 允许 第 0 天 (实际 上 返回 的 
日 期 是 上 一 个 月 的 最 后 一 天 ) ,而 0 月 表示 十 二 月 ,0 年 表示 2000 年 。 

3. 验证 URL 

如 果 表 单 里 请 求 用 户 输 入 其 主 站 地 址 ,那么 就 需要 对 这 个 URL 进行 检验 来 判断 它 
是 否 像 个 有 效 地 址 。 对 URL 的 检验 也 可 以 使 用 正则 表达 式 , 可 以 使 用 eregi() 函 数 。 
ereg() 与 eregi() 之 间 唯 一 的 区 别 是 ,ereg() 对 模式 区 分 字母 大 小 写 , 而 eregi() 则 不 区 分 
大 小 写 。 以 下 展示 了 URL 的 标准 化 与 检验 的 代码 范例 : 

eregi ("^((httplhttps|ftp) ://)? ([[:alnom:]\- \.])+ (\.) ([[:alnum:]]){2,6} ([[:alnum:]/+=%& 

_\.~2\-]*)$",Swebsite); 

为 了 验证 URL ,首先 检查 可 选 的 http://、https:// 或 ftp://, 然 后 希望 看 到 字母 , 数 
字 或 下 划 线 ,后 面 接着 一 个 英文 句点 ,再 接着 2 一 6 个 字母 的 字符 串 (com、edu 等 )。 最 后 
允许 可 能 出 现 的 许多 其 他 字符 ,它们 将 构成 特定 的 文件 名 以 及 要 发 送 给 它 的 参数 等 。 

4. 魔术 引用 magic_quotes_gpe 

PHP 中 构建 了 一 个 方便 的 特性 , 称 为 Magic Quotes( 魔 术 引 用 )。 在 PHP 中 ,有 两 类 
主要 的 Magic Quotes: magic_quotes_gpc, 它 适用 于 表单 .URL 和 cookie 数据 (Cgpc 代表 
get、post、cookie) ;magic_quotes_runtime, 它 适用 于 从 外 部 文件 和 数据 库 检 索 的 数据 。 

PHP 提供 magic_quotes_gpc 魔法 引用 功能 来 保护 网 站 系统 免 受 攻击 , 它 会 自动 从 用 
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户 的 输入 串 中 过 滤 特 殊 字符 (\"\\\0O CNULL)) ,并 减 慢 输入 的 过 程 。 在 处 理 数 据 库 或 
HTML 时 ,这 有 助 于 防止 出 现 问 题 。 但 是 ,如 果 往 表单 中 输入 一 些 文本 ,其 中 包含 一 个 单 
引号 (7 ,如 图 6-12 所 示 ,在 重新 打印 这 段 文本 时 ,得 到 的 页 面 看 起 来 会 很 奇怪 ,如 图 6-13 
所 示 。 


OL Er | 
请 输入 您 的 信息 : 

姓名 Mary 

Email: nary163@126. con 

性 别 @@ 男 @ 女 

年 龄 50 内 ~ 


Nary’s honel 


个 人 简介 


谢谢 您 Eary， 您 的 信息 如 下 : 
Email: ey1630126. com 


-29 
Fa Jary\V s home! 
|| 我 们 将 回复 您 至 mary7638736. com 


图 6-13 ”输入 到 表单 中 的 撤 号 会 被 PHP 自动 进行 转 义 ,从 而 生成 不 合适 的 结果 


如 果 在 服务 器 上 启用 了 Magic Quotes, 则 可 以 使 用 stripslashes() 函 数 撤销 它 的 
作用 。 


$var= stripslashes ($ var); 


这 个 函数 将 删除 在 $ var 中 发 现 的 任何 反 斜 杠 。 在 表单 示例 中 ,这 个 函数 起 到 把 已 
转 义 的 提交 字符 串 转变 回 其 原来 的 未 转 义 值 的 作用 。 

PHP 提供 这 个 参数 的 初 囊 是 好 的 ,但 是 该 功能 从 出 现 以 来 一 直 备 受 争 议 。 实 践 经 验 
表明 ,如果 采用 magic_quotes_gpc, 与 不 使 用 该 函数 相 比 .需要 两 倍 多 的 内 存 来 处 理 每 条 
输入 的 元 素 , 因 此 ,如 果 非 必要 ,可 以 在 php. ini 文件 把 该 参数 设置 为 关闭 ,不 使 用 该 功 
能 , 转 用 其 他 的 方法 来 处 理 。 
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可 以 使 用 与 stripslashes() 函 数 对 立 的 函数 addslasher() ,来 模拟 禁用 Magic Quotes 
时 它 所 做 的 工作 。 

另外 , 当 处 理 源 于 表单 的 字符 串 时 ,使 用 trim() 函 数 也 是 一 个 好 主意 , 它 会 从 值 的 两 
端 删除 多 余 的 空白 。 


$name=trim($ name); 


5. 处 理 HTML 输入 

用 户 有 时 可 能 在 发 送 给 浏览 器 的 文本 中 做 入 HTML 元 素 ,需要 提防 这 一 点 ,如 果 在 
Web 页 面 中 显示 该 文本 , 它 可 能 包括 恶意 脚本 ,该 脚本 会 关闭 Web 页 面 ,或 者 导航 到 另 
一 个 页 面 。 

因此 ,用 户 可 以 轻松 地 添加 HTML 或 JavaScript 代码 到 表单 数据 中 ,比如 表单 中 的 
多 行文 本 域 。 

许多 动态 驱动 的 Web 应 用 程序 会 获取 用 户 提 交 的 信息 ,将 其 存储 在 数据 库 中 ,然后 
把 该 信息 重新 显示 在 另 一 个 页 面 上 。 例 如 ,在 发 布 新 闻 ,博文 或 注册 表单 时 ,如 果 用 户 在 
他 们 的 输入 框 中 输入 HTML 代码 ,这 段 代码 可 能 抛弃 站 点 的 布局 和 美感 。 更 糟糕 的 是 ， 
不 良 代 码 可 能 创建 弹出 式 窗 口 或 者 重 定向 到 其 他 站 点 ,如 图 6-14 所 示 , 不 怀 好 意 的 用 户 
可 能 输入 非法 代码 ,这 是 一 种 常见 的 攻击 方法 。 


(POG ro /orto o- cx Snesm >* 因 
请 输入 您 的 信息 ， 
姓名 : Jary 


Email: mary1639126. con 


性 别 目 男 回 女 


年 龄 30 内 ~ 


<script language=" javascript">alert( 这 和 
是 广告 + *);《/script>》 
个 人 简介 


图 6-14 恶意 用 户 将 JavaScript 代码 输入 到 文本 输入 框 中 


PHP 包含 几 个 函数 ,用 于 处 理 在 字符 串 内 发 现 的 HTML 及 其 他 代码 。 
(1) 把 特定 字符 转化 为 HTML 实体 。 


$htmlready=htmlspecialchars ($ string, $ quote style); 


一 些 常 用 字符 在 HTML 中 有 特殊 含义 ,它们 被 转化 为 相应 的 HTML 实体 ,从 而 避 
免 作为 HTML 代码 被 解释 。 被 转化 的 字符 包括 和 号 (&&) 、 双 引号 (") 或 单 引 号 () (取决 
于 $ quote_style 的 设置 )、 小 于 号 (二 ) 和 大 于 号 (二 )。 
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(2) 把 全 部 字符 转化 为 相应 的 HTML 实体 。 
$htmlready=htmlentities ($ string); 


任何 包含 HTML 实体 的 字符 都 被 转化 为 对 应 的 实体 。 
(3) 删除 字符 串 里 的 HTML 标记 。 


$clean= strip tags($ string, $allow); 


字符 串 里 全 部 HTML 标记 或 PHP 标记 都 被 删除 ,但 可 选 参数 $ allow 里 指定 的 标 
记 除 外 。 
(4) 把 新 行 符 转化 为 HTML 分 隔 符 。 


$htmlready=nl12br ($ string); 


得 到 的 字符 串 在 所 有 的 新 行 之 前 都 包含 “二 br /二 ”。 
6.3.3 设计 过 程 
(1) 在 文本 编辑 器 中 修改 脚本 6-5( 参 见 脚本 6-6) 。 
脚本 6-6 在 脚本 6-5 的 基础 上 ,检查 Email 日 期 `URL 的 有 效 性 ,并 处 理 HTML 输入 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DID XHTML 1.0 Transitional//EN" 

2 "http://www.w3.org/TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

3 “<“html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv="Content- Type" content="text/html; charset=gb2312" /> 
6 <title> 注 册 表 单 </title> 

7 </head> 

8 <body> 

9 <p align= "center"> 用 户 注 册 <br /> * 号 标志 必 填 < /p> 

10 < form name= "forml" method= "post" action=""> 


60 </form> 


62 <?php #Script 6- 7- handle regist.php 
63 if($_POST['Submit"']){ 

64 if(empty($_POST['name'])){ 
65 echo "用 户 名 不 能 为 空 ."; 
66 exit; 

67 } 

68 if(empty($_POST["passl"])){ 
69 echo "密码 不 能 为 空 。"; 
70 exit; 

1 } 

骆 if (empty($_POST['pass2°])){ 
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echo "确认 密码 不 能 为 空 。"; 
exit; 
} 
if(empty($_POST['email'])){ 
echo "Email 不 能 为 空 。"; 
exit; 
} 
if($_POST['pass1']!==$_POST['pass2']){ 
echo "密码 与 确认 密码 不 同 ,请 重新 输入 。"; 
exit; 
} 
if(!ereg("^([a-zA-20-9 -])+8 ([a-zA-20-9 -])+(\.[a-zA-20-9 ~-]{2,6}))+", 
$_FOST['email'])){ 
echo "Email 格式 不 正确 ."; 
exit; 
} 
$output= strtotime ($_POST['birthday']); 
if ($output===false) { 
echo "日 期 格式 不 正确 "; 
exit; 
} 
if(!eregi("^((httplhttps|ftp) ://)? ([[:alnum:] \-\.])+ (\.) ([[:alnum:]]) {2,6} 
([[:alnum:]/+=%& \.~?\-]*)$",$_POST['web'])){ 
echo "主页 格式 不 正确 ."; 
exit; 


?> 
<p align= "center"> 您 的 输入 为 :< /p> 
<table width= "500"> 
<tr> 
<td> 用 户 名 < /td><td><?php echo $_POST['name'];?>< /td> 
</tr> 
> 
<td> 密 码 < /td><td><?php echo $_POST['pass1'];?>< /td> 
</tr> 
<tr> 
<td>Email 地 址 < /td><td><?php echo $_POST['email'];?>< /td> 
</tr> 
<tr> 
<td> 性 别 < /ta> 
<td> 
<?2php 
switch ($_POST["gender']){ 


case 1: 


135. 
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115 echo " 男 "; 

116 break; 

117 Case 2: 

118 echo " 女 "; 

119 break; 

120 case 3: 

121 echo "保密 "; 

122 break; 

123 } 

124 ?> 

125 </td> 

126 </tr> 

127 <tr> 

128 <td> 出 生日 期 < /td> 

129 <td> 

130 <?php 

131 echo date('Y 年 m 月 d 日 ',$ output); 

132 ?> 

133 </td> 

134 </tr> 

135 <tr> 

136 <td> 爱 好 < /td> 

137 <td> 

138 <?php 

139 foreach ($ interests as $value){ 

140 echo $value; 

141 echo "gnbsp; gnbsp;"; 

142 } 

143 ?> 

144 </td> 

145 </tr> 

146 <tr> 

147 <td> 主 页 < /td> 

148 <td><a href=<?php echo $_POST['web'];?>><?php echo $_POST['web'];?></a> 
</td> 

149 </tr> 

150 <tr> 

151 <td> 个 人 简介 < /td> 

152 <td> 

153 <?php 

154 S$ more= stripslashes ($_POST['more']); 

155 $more=htmlspecialchars ($more); 

156 S$ more= str_replace (" ", "gnbsp;",$ more); 


157 echo nl2br ($ more); 
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158 ?></td> 
159 </tr> 
160 </table> 
161 <?php 

162 } 

163 ?> 

164 < /body> 

165 < /html> 


(2) 在 五 个 让 语句 后 添加 验证 电子 邮件 地 址 。 


if(!ereg("^([a- zA- 20-9 -1])+@ ([a-zA- 20-9 -])+(\.[a-zA-20-9 -]{2,6})+", $_POSTI[' 
email'])){ 

echo "Email 格式 不 正确 。"; 

exit; 


} 
(3) 添加 验证 日 期 格式 。 


$output=strtotime ($_POST['birthday']); 
if ($output===false){ 

echo "日 期 格式 不 正确 "; 

exit; 


} 
(4) 添加 验证 主页 URL。 


if(!eregi("^((httplhttps|ftp)://)? ([[:alnom:] \-\.])+ (\.) ([[:alnom:]]) {2,6} ([[:alnum:]/ 
+=%& \.~?\-]*)$",$_POST['web'])){ 

echo "主页 格式 不 正确 。"; 

exit; 


} 
(5) 出 生日 期 显示 的 部 分 ,使 用 date() 函 数 , 格 式 化 显示 日 期 。 
echo date("'Y 年 m 月 d 日 ',$output); 
(6) 个 人 简介 的 多 行文 本 框 显示 ,需要 经 过 一 系列 的 HTML 输入 处 理 。 


$ more= stripslashes ($_POST['more']); 
$more=htmlspecialchars ($more); 
$ more=str_replace(" ", "gnbsp;", $ more)’; 


echo nl2br ($more); 


首先 使 用 stripslashes() 函数 删除 魔术 引用 产生 的 反 斜 杠 ,然后 使 用 htmlspecialchars() 


函数 对 各 种 特殊 字符 转变 成 HTML 实体 格式 .因此 任何 双 引号 都 会 被 转变 为 "& quot; ”， 


“二” 和 “二 ” 则 分 别 变 成 *&1t;” 和 *&gt;”。 


然后 使 用 str_replace() 转 换 空格 字符 .使 用 nl2br() 函 数 把 回 车 符 转变 成 一 个 
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HTML 的 “一 br /二 ”标签 。 
(7) 其 他 部 分 代码 不 变 , 将 该 页 另存 为 handle_regist. php ,并 在 Web 浏览 器 中 测试 
它们 (参见 图 6-15 和 图 6-16) 。 


男 ” © 女 时 保密 


1/ 属 式 1900-01-01 
国电 影 回音 乐 回 运 动 加 看 书 回 其 他 


http:// 


6-16 ”多 行 输入 框 中 的 特殊 字符 及 输入 格式 都 被 正常 显示 


htmlspecialchars() 和 htmlentities() 函 数 都 可 以 带 一 个 可 选 参 数 , 指 示 应 该 如 何 处 理 
引号 。 查 阅 PHP 帮助 手册 ,可 了 解 特定 细节 。strip_tags() 函数 甚至 会 清除 无 效 的 
HTML 标签 ,这 些 标签 可 能 会 引发 问题 。nl2br() 函数 是 一 个 与 安全 无 关 但 相当 有 用 的 
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了 数 。 

同时 ,依据 经 验 , 仅 当 使 用 任何 其 他 函数 或 技术 都 不 能 完成 手头 的 任务 时 , 才 应 该 使 
用 正则 表达 式 ,尽管 这 个 示例 演示 了 如 何 编写 和 执行 自己 的 正则 表达 式 , 但 是 实际 需要 
时 ,搜索 因特网 可 以 找到 众多 已 经 可 用 的 示例 。 


本 章 小 结 


本 章 讨 论 了 在 PHP 环境 中 表单 的 基本 构成 .原理 ,并 且 对 用 户 表单 提交 的 处 理 方 
法 ,数据 验证 等 做 了 详细 讲述 。 


重点 回顾 
1. 如 何 创建 表单 及 表单 控件 元 素 。 
2. 使 用 超 全 局 变量 $_POST 和 $ _GET 获取 表单 输入 。 
3. 开发 安全 有 效 的 代码 ,验证 表单 数据 。 
本 章 实 训 
【 实 训 1 


新 建 login_get. html 和 handle_get. php 两 个 页 面 ,在 login_get. html 中 添加 注册 表 
单 ,效果 如 图 6-17 所 示 ,将 该 表单 的 数据 使 用 get 方法 提交 给 handle_get. php 页 面 。 


图 6-17 login_get. html 的 运行 效果 


【 实 训 2】 

新 建 login_post. php 页 面 ,添加 注册 表单 ,效果 同 实 训 1, 将 该 表单 的 数据 使 用 post 
方法 提交 ,在 login_post. php 这 个 页 面 中 分 别 完成 显示 表单 和 处 理 表单 ,要 求 未 提交 表单 
时 显示 表单 ,提交 表单 后 不 再 显示 表单 ,只 显示 提交 的 内 容 。 
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【 实 训 3】 

将 第 3 章 实 训 2 的 99. php 另存 为 nn. php， 
实现 NN 乘法 表 , 将 算法 写成 函数 ,并 通过 文本 
框 输入 的 变量 传递 参数 ,验证 是 否 输 入 为 数字 
(提示 : 使 用 intval 函数 ) ,验证 正确 后 将 结果 输 
出 为 表格 格式 。 网 页 运行 效果 如 图 6-18 所 示 。 

【 实 训 4】 

新 建 submit. php, 完 成 表单 综合 开发 ,具体 

(1) 表单 界面 设计 结合 本 章 的 脚本 6-5 及 
第 3 章 的 脚本 3-2, 将 生日 设计 为 下 拉 菜单 ,如 图 6-19 所 示 。 


图 6-18 nn. php 的 运行 效果 


注册 

基本 信息 (* 为 作 十 厅 ) 
用 户 名 * 

9 


无 安全 提问 


| 见 [C| 女 |C| 保 宫 


W070 -年 1 -月 1 -日 


图 6-19 ”submit. php 的 运行 效果 


(2) 将 显示 表单 与 处 理 表单 的 代码 写 在 同一 个 php 文件 中 ,验证 表单 是 否 提交 。 
(3) 验证 必 填 项 目 是 否 为 空 , 验 证 两 次 密码 是 否 相同 。 

(4) 验证 email 和 URL 的 有 效 性 。 

(5) 有 效 处 理 HTML 输入 。 


第 7 章 创建 动态 Web 站 点 


在 掌握 PHP 的 基本 知识 之 后 ,就 可 以 开始 构建 真正 的 动态 Web 站 点 。 与 最 初 的 静 
态 Web 站 点 相 比 ,动态 Web 站 点 更 容易 维护 ,能 更 快 地 对 用 户 做 出 响应 ,并 且 内 容 能 够 
改变 以 响应 不 同 的 情况 。 本 章 讨论 使 用 PHP 来 进行 文件 处 理 。 在 Web 应 用 程序 中 ,将 
数据 存储 在 服务 器 上 特别 有 用 ,因为 这 样 可 以 使 数据 持久 保存 。 博 客 .客户 名 册 、 反 馈 页 
面 都 需要 使 用 服务 器 上 的 文件 。 本 章 将 介绍 三 个 新 应 用 ,它们 都 常用 于 创建 更 复杂 的 
Web 应 用 程序 。 


二 包 合 渐 个 天 忻 


到 此 为 止 ,本 书 中 的 每 个 脚本 都 由 单个 文件 组 成 . 它 包含 所 有 需要 的 HTML 和 PHP 
代码 。 但 是 ,在 开发 更 复杂 的 Web 站 点 时 ,将 看 到 这 种 方法 有 很 多 局 限 性 。PHP 可 以 很 
容易 地 利用 外 部 文件 ,从 而 能 够 把 脚本 分 成 各 个 不 同 的 部 分 。 我 们 可 以 频繁 地 使 用 外 部 
文件 ,从 PHP 中 提取 HTML .或 者 分 离 出 常用 的 过 程 。 


7.1.1 包含 外 部 文件 函数 


在 没有 介绍 包含 外 部 文件 功能 之 前 , 当 创建 了 一 个 真正 有 用 的 函数 ,只 能 选择 将 这 个 
函数 粘贴 到 需要 使 用 它 的 每 个 文档 中 。 如 果 发 现 一 个 漏洞 (bug) 或 者 想 要 添加 一 项 功 
能 ,也 必须 找到 包含 的 每 个 页 面 并 且 修 改 它 ,一 次 又 一 次 地 重复 。 若 使 用 外 部 包含 函数 ， 
则 不 必 这 么 烦琐 。 可 以 把 新 创建 的 函数 添加 到 一 个 单独 的 文档 中 ,例如 myinclude. php， 
然后 ,在 运行 的 时 候 ,将 其 读 入 到 需要 它 的 任何 页 面 。 

PHP 有 4 个 用 于 外 部 文件 的 函数 : include() include_once()、require() 和 require_ 
once()。 这 些 函 数 能 够 把 其 他 文件 ,通常 是 其 他 PHP 脚本 包含 到 PHP 文档 中 。 这 些 包 
含 的 文件 中 的 PHP 代码 就 好 像 是 主 文档 的 一 部 分 一 样 来 执行 ,这 对 于 在 多 个 页 面 中 包 
含 代码 库 是 很 有 用 的 。 

这 4 个 函数 的 用 法 相同 ,PHP 脚本 中 将 包括 如 下 代码 行 : 


include once('filename.php'); 


require('/path/to/filename.htm]l'); 
使 用 其 中 任何 一 个 函数 的 最 终结 果 都 是 ,获取 包含 文件 (included file) 的 所 有 内 容 ， 


142 


PHP+MySQL 项 目 实例 开发 


并 在 那 一 刻 从 父 脚 本 (调用 该 函数 的 脚本 ) 中 删除 该 文件 。 包 含 文件 的 一 个 重要 属性 是 ， 
PHP 将 把 包含 代码 视 作 HTML( 即 直接 把 它 发 送 到 浏览 器 ) ,除非 它 包含 PHP 标签 内 的 
代码 。 

从 功能 上 讲 , 包 含 文件 使 用 什么 扩展 名 无 关 紧 要 , 它 可 以 是 . php 或 . html。 不 过 , 通 
过 给 文件 提供 一 个 象征 性 的 名 称 有 助 于 传达 其 目的 (例如 , HTML 的 包含 文件 可 能 使 用 
.inc, 或 . html) 。 


7.1.2 绝对 路 径 与 相对 路 径 


在 引用 任何 外 部 项 目 ( 它 可 以 是 PHP 中 的 包含 文件 .HTML 中 的 CSS 文档 或 者 图 
像 ) 时 ,可 以 选择 使 用 绝对 路 径 或 相对 路 径 。 绝 对 路 径 从 计算 机 的 根 目 录 开始 指出 文件 的 
位 置 。 不 管 引 用 ( 父 ) 文 件 的 位 置 是 什么 ,这 种 路 径 总 是 正确 的 。 例 如 ,PHP 脚本 可 以 使 
用 如 下 代码 包含 文件 : 

include ('C:/php/includes/file.php'); 

include ('/usr/xyz/includes/file.php'); 

假定 file. php 存在 于 指定 的 位 置 , 这 种 包含 方式 将 会 有 效 ( 除 非 有 任何 许可 或 权限 问 
题 )。 上 述 两 个 分 别 是 Windows 系统 下 和 类 UNIX 系统 下 的 绝对 路 径 , 绝 对 路 径 总 是 以 
像 C:/ 或 /这 样 的 内 容 开 头 。 

相对 路 径 使 用 引用 ( 父 ) 文 件 作为 起 点 。 要 移 到 上 一 级 文件 夹 ,可 以 一 起 使 用 两 个 名 
点 。 要 移 到 一 个 文件 夹 中 ,可 使 用 其 名 称 后 面 接着 一 个 斜 杜 。 假 定 当 前 脚本 位 于 www/ 
exl 文件 夹 中 ,如 果 想 在 www/ex2 中 包含 一 些 内 容 , 代 码 如 下 : 

include ('../ex2/file.php'); 

相对 路 径 将 保持 准确 ,即使 移 到 另 一 个 服务 器 上 ,只 要 文件 保持 它们 的 当前 关系 
即 可 。 

7.1.3 include() 和 require() 的 区 别 

include() 和 require() 函 数 在 正确 工作 时 完全 一 样 ,但 是 在 它们 失败 时 会 表现 得 有 所 
不 同 。 如 果 include() 函 数 不 工 作 ( 由 于 某 种 原因 而 不 能 包含 文件 ) ,就 会 向 Web 浏览 器 
打印 一 个 警告 ,如 图 7-1 所 示 , 但 是 脚本 会 继续 运行 。 如 果 require() 失 败 ,就 会 打印 一 个 
错误 ,并 且 脚 本 会 中 止 运行 ,如 图 7-2 所 示 。 

这 两 个 函数 还 有 一 个 *_onceO) 版 本 ,它们 保证 处 理 的 文件 只 会 被 包含 一 次 ,而 不 管 
脚本 可 能 (假定 不 经 意 地 ) 试 图 包含 它 多 少 次 。 


require once('filename.php'); 


include once('filename.php'); 


7.1.4 站 点 文件 结构 
在 Web 应 用 程序 中 使 用 多 个 文件 时 ,总 体 站 点 结构 将 变 得 更 重要 。 在 对 站 点 进行 布 
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EEC ET 且 


Warning: include(file.php) [function include]: failed to open stream: No 
such file or directory in C:NAppServNwwwNbook\ch07\include. php on line 
18 


Warning: include() [function. include]: Failed opening "file.php’” for 


inclusion (include path=’.;C:\php5\pear’ ) in C:\AppServ\www\book\ch07 
Ninclude. php on line 18 


欢迎 来 到 我 的 主页 
PHP 有 4 个 用 于 外 部 文件 的 函数 ，include() 、include_once() 、requireO) 和 


require_once()。 


这 些 函 数 能 够 把 其 他 文件 ， 通 常 是 其 他 PHP 脚 本 包含 到 PHP 文 档 中 。 这 些 包含 的 文件 中 的 
PE 这 对 于 在 多 个 页 面 中 包含 代码 库 是 很 有 


这 4 个 函数 的 用 法 相同 ，PHP 脚 本 中 将 包括 如 下 代码 行 ， 


include_once( filename. php ) 


require( /path/to/filename. html’ ) 


图 7-1 include() 调 用 失败 生成 这 2 条 警告 ,但 是 余下 的 页 面 会 继续 执行 


OB eatensooer0 p- cxl|soiaaxt > 国医 


Warning: require(file.php) [function require]: failed to open stream: No 
such file or directory in C:VAppServNwww\book\ch07\require. php on line 
18 

Fatal error: require() [function. require]: Failed opening required 
"file.php’ (include_path= , ;C:\php5\pear’ ) in C:\AppServ\www\book\ch07 
\require. php on line 18 


图 7-2 ”require() 函 数 调用 失败 将 会 打印 一 个 致命 错误 ,并 中 止 脚本 的 执行 


局 时 ,需要 考虑 3 个 事项 : 

。 易 于 维护 。 

。 安全 性 。 

。 便于 用 户 导航 。 

使 用 外 部 文件 保存 核心 代码 ( 即 PHP 代码 )、CSS、JavaScript 和 HTML 设计 ,将 极 大 
地 改进 维护 站 点 的 简易 性 .因为 共同 编辑 的 代码 都 放置 在 一 个 中 央 位 置 。 例 如 建立 
includes 或 templates 目录 来 存储 这 些 文件 ,将 其 与 主要 的 脚本 (直接 在 Web 浏览 器 中 访 
问 它们 ) 隔 离开 。 

建议 为 安全 性 不 成 问题 的 文档 (如 HTML 模板 ) 使 用 . inc 或 . html 文件 扩展 名 ,为 那 
些 包含 更 多 敏感 数据 (如 数据 库 访问 信息 ) 的 文档 使 用 . php 扩展 名 。 也 可 以 同时 使 用 
.inc 和 . html 或 . php, 以 将 一 个 文件 明显 地 指定 为 某 种 类 型 的 包含 文件 ,如 db. inc. php 
或 header. inc. html。 

最 后 ,尽力 将 站 点 构造 成 便于 用 户 通过 单 击 链接 以 及 手动 输入 URL 来 进行 导航 的 
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形式 。 尽 量 避 免 创建 过 多 的 腐 套 文件 夹 ,或 者 使 用 难以 输入 的 、 包 含 大 小 写字 母 以 及 各 种 
标点 符号 的 目录 名 和 文件 名 。 

在 下 一 个 示例 中 ,将 使 用 包含 文件 把 HTML 格式 化 代码 与 PHP 代码 隔 开 。 这 样 ， 
这 些 页 面 将 具有 相同 的 外 观 , 就 好 像 它 们 都 是 相同 Web 站 点 的 一 部 分 一 样 ,而 不 必 每 次 
都 重 写 HTML 代码 。 这 种 技术 创建 了 一 个 模板 系统 ,这 是 使 大 型 应 用 程序 具有 一 致 性 
和 可 管理 性 的 一 种 容易 的 方式 。 这 些 示 例 关 注 的 焦点 是 PHP 代码 本 身 。 

(1) 在 文本 编辑 器 或 所 见 即 所 得 编辑 器 中 设计 一 个 HTML 页 面 (参见 脚本 7-1 和 
图 7-3) 。 


脚本 7-1 Web 页 面 的 HTML 模板 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.org/TR/xhtml1/DTD/xhtmll-transitional.dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv="Content- Type" content="text/html; charset=gb2312" /> 
6 ”<title> 欢 迎 光 临 < /title> 

7 <linkhref="includes/layout.css" rel="stylesheet" type="text/css" /> 
8 </head> 

9 

10 <body> 

11 <div id="wrapper"> 

12 <div id= "content"> 

13 <div id= "nav"> 

14 <h3> 请 选择 < /h3> 

15 <ul> 

16 <1i class= "navtop"><a href= "index.php" title= "首页 "> 首页 < /a>< /1i> 
17 <1i><a href= "blog.php" title= "博客 "> 博客 </a></1i> 

18 <1li><ahref= "images.php" title=" 相 册 "> 相 册 < /a>< /1i> 
19 <1li><a href="register.php" title= "注册 "> 注册 < /a></1i> 
20 </ul> 

21 </div> 

22 

23 <hl id- "mainhead"> 父 标题 < /hl> 

24 <p> 主 要 内 容 < /p> 

25 <p> 主 要 内 容 < /p> 

26 <p> 主 要 内 容 </p> 

27 <p> 主 要 内 容 </p> 

28 <h2> 子 标题 < /h2> 

29 <p> 主 要 内 容 < /p> 

30 <p> 主 要 内 容 < /p> 

31 <p> 主 要 内 容 < /p> 


32 <p> 主 要 内 容 < /p> 
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33 </div> 

34 

35 <div id= "footer"><p> &copy; Copyright 2014< /p>< /div> 
36 </div> 

37 

38 </body> 

39 </html> 


meen PX 
请 选择 


© Copyright 2014 


7-3 未 使 用 任何 PHP 代码 的 HTML 与 CSS 设计 


要 开始 为 Web 站 点 创建 一 个 模板 ,可 以 像 标准 HTML 页 面 那样 设计 布局 , 它 独 立 于 
任何 PHP 代码 。 

为 了 节省 篇 幅 , 此 处 不 再 给 出 本 示例 的 CSS 文件 ( 它 控制 布局 ) 。 你 可 以 不 使 用 该 文 
件 ,模板 仍 会 工作 ,只 是 它 看 起 来 不 那么 美观 。 

(2) 复制 从 布局 源 代码 的 第 一 行 到 紧 接 着 特定 于 页 面 的 内 容 之 前 的 所 有 内 容 , 并 将 
其 粘贴 到 一 个 新 文档 中 (参见 脚本 7-2) 。 


脚本 7-2 每 个 Web 页 面 的 初始 HTML 代码 将 存储 在 头 文件 中 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

"http://www-w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 

<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 

<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title><?php echo $ page title;?></title> 

< link href="includes/layout .css" rel="stylesheet" type="text/css" /> 

< /head> 
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9 
10 <body> 
11 <div id="wrapper"> 


12 <div id= "content"> 
13 <div id= "nav"> 

14 <h3> 请 选择 < /h3> 

15 <ul> 

16 <1i class="navtop"><a href="index.php" title=" 首 页 "> 首页 </a></1i> 
17 <1i><a href= "blog.php" title= "博客 "> 博客 </a></1i> 

18 <1i><a href="images.php" title= "相册 "> 相册 < /a>< /1i> 

19 <1i><a href="register.php" title=" 注 册 "> 注 册 < /a>< /1i> 

20 </ul> 

21 </div> 

22 < !-—script7- 2- header .html> 


第 一 个 文件 将 包含 初始 HTML 标签 .从 DOCTYPE 到 页 面 主体 的 开始 处 , 它 还 具有 
用 于 在 浏览 器 窗口 右 侧 建立 链接 列 的 代码 ,如 图 7-3 所 示 。 
(3) 更 改 页 面 的 标题 行 , 以 便 读 取 。 


<title><?php echo $page title; ?></title> 


页 面 的 标题 对 于 各 个 页 面 是 可 变化 的 。 为 了 做 到 这 一 点 ,把 它 设置 成 一 个 变量 ,通过 
PHP 将 其 打印 出 来 。 

(4) 将 文件 另存 为 header. html。 

包含 文件 可 以 为 文件 名 使 用 几乎 任何 扩展 名 。 有 些 程序 员 喜 欢 使 用 . inc 指示 一 个 文 
件 将 被 用 作 包 含 文件 。 在 这 种 情况 下 ,也 可 以 使 用 . inc. html, 这 指示 它 既 是 一 个 包含 文 
件 , 又 是 一 个 HTML 文件 ,将 其 与 全 部 由 PHP 代码 组 成 的 包含 文件 区 分 开 。 

(5) 复制 原始 模板 中 从 特定 于 页 面 的 内 容 末 尾 到 页 面 末尾 的 所 有 内 容 ,并 将 其 粘贴 
到 一 个 新 文件 中 (参见 脚本 7-3) 。 


脚本 7-3 用 于 每 个 Web 页 面 的 最 终 HTML 代码 将 存储 在 这 个 页 脚 文件 中 。 


1 <!--Script 7-3 - footer.html -一 > 

2 <div id= "footer"><P> &copy; Copyright 2014< /p>< /div> 
3 </div> 

4 

5 </body> 

6 </htm> 


页 脚 文件 包含 用 于 页 面 主体 的 余下 格式 化 代码 ,包括 页 面 的 版 权 信息 日 期 等 页 脚 信 
息 ,之 后 关闭 HTML 文档 。 

(6) 将 文件 另存 为 footer. html。 

(7) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 7-4) 。 
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脚本 7-4 这 个 脚本 使 用 存储 在 外 部 文件 中 的 模板 生成 一 个 完整 的 Web 页 面 。 


1 <?php#Script 7- 4 -index.php 
$page_title= ' 欢 迎 光临 !'; 

3 include ('./includes/header.html '); 
4 ?> 

5 <hl id= "mainhead"> 父 标题 < /hl> 
6 <p> 主 要 内 容 < /p> 

7 < 户主 要 内 容 < /p> 

8 <p> 主 要 内 容 < /p> 

9 <p> 主 要 内 容 < /p> 

10 <h2> 子 标题 < /h2> 

11 <p> 主 要 内 容 < /p> 

拉 <p> 主 要 内 容 < /p> 

13 <p> 主 要 内 容 < /p> 

14 <p> 主 要 内 容 < /p> 

15 <?php 

16 include('./includes/footer.html'); 
17 ?> 


因为 这 个 脚本 将 为 其 大 部 分 HTML 格式 化 代码 使 用 包含 文件 ,所 以 可 以 使 用 PHP 
标签 (而 不 是 HTML 标签 ) 来 开始 和 结束 文件 。 
(8) 设置 $ page_title 变量 .并 且 包 含 HTML 头 文件 。 


$ page title= "Welcome to this Site!'; 
include ('./includes/header.html'); 


$ page_title 将 允许 我 们 为 使 用 这 个 模板 系统 的 每 个 页 面 设置 一 个 新 的 标题 。 因 为 
在 包含 头 文件 之 前 建立 了 这 个 变量 ,所 以 头 文件 将 能 够 访问 它 。 记 住 : include() 这 一 行 
具有 把 包含 文件 的 内 容 放 入 该 页 面 中 这 个 位 置 的 作用 。 

(9) 关闭 PHP 标签 ,并 从 模板 上 复制 特定 于 页 面 的 内 容 。 


2> 
<hl id= "mainhead"> 父 标题 < /hl> 
<p> 主 要 内 容 < /p> 
<p> 主 要 内 容 < /p> 
<p> 主 要 内 容 < /p> 
<p> 主 要 内 容 < /p> 
<h2> 子 标题 < /h2> 
<p> 主 要 内 容 < /p> 
<p> 主 要 内 容 < /p> 
<p> 主 要 内 容 < /p> 
<p> 主 要 内 容 < /p> 


可 以 使 用 echo() 把 这 段 信息 发 送 给 浏览 器 ,但 是 因为 这 里 没有 动态 内 容 , 暂 时 退出 
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PHP 标签 将 会 使 操作 更 容易 、 更 高 效 。 

(10) 创建 最 终 的 PHP 部 分 ,并 包含 页 脚 文件 。 

<?Pphp 

include (' ./includes/footer.html'); 

2 

(11) 将 文件 另存 为 index. php, 并 上 传 到 Web 服务 器 。 

(12) 在 与 index. php 相同 的 文件 夹 中 创建 一 个 includes 目录 。 然 后 把 header 
. html ,footer. html 和 layout. css 这 几 个 文件 放 到 这 个 目录 中 。 

(13) 通过 Web 浏览 器 进入 index. php 页 面 ,来 测试 模板 系统 (参见 图 7-4) 。 


© Copyright 2014 


图 7-4 现在 使 用 PHP 中 的 外 部 文件 创建 了 相同 的 布局 


index. php 页 面 是 这 个 模板 系统 的 最 终结 果 ,我们 不 必 直 接 访问 任何 包含 文件 ,因为 
index. php 将 负责 纳入 它们 的 内 容 。 

在 php.ini 文件 中 ,可 以 调整 include_path 设置 , 它 指示 是 否 允 许 PHP 检索 包含 
文件 。 

一 种 最 佳 实践 是 , 当 引 用 与 父 ( 包 含 ) 文 件 相 同 目录 内 的 文件 时 ,可 使 用 . /filename 语 
法 。 存 储 在 父 文件 之 上 目录 中 的 文件 将 使 用 路 径 . . /filename 来 包含 ,存储 在 父 文件 之 下 
目录 中 的 文件 将 使 用 . /directory/filename 来 包含 。 

可 以 使 用 相对 路 径 或 绝对 路 径 包含 文件 


include ('/path/to/filename'); 
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7.2 PHP 上 传 文件 的 原理 与 实现 


7.2.1 利用 PHP 上 传 文件 


文件 的 上 传 也 是 Web 应 用 程序 文件 操作 中 一 个 很 重要 的 组 成 部 分 。 因 为 大 部 分 文 
件 存在 于 用 户 客户 端的 机 器 上 ,如 果 要 想 让 其 他 用 户 访问 则 必须 要 把 它 上 传 到 服务 器 。 
如 图 片 网 站 上 传 图 片 、 软 件 下 载 站 需要 上 传 软件 ,音乐 站 点 需要 上 传 MP3 音乐 文件 等 。 
PHP 也 提供 了 对 文件 上 传 的 支持 。 

与 使 用 PHP 处 理 任何 HTML 表单 一 样 ,上 传 文件 有 两 个 过 程 : 首先 ,必须 显示 
HTML 表单 ,用 正确 的 代码 运行 文件 上 传 ;然后 ,在 提交 表单 时 ,PHP 脚本 必须 把 上 传 的 
文件 复制 到 正确 的 路 径 。 

为 了 让 这 个 过 程 顺利 进行 ,必须 做 好 以 下 的 准备 工作 : 

。 必须 运行 具有 正确 设置 的 PHP。 

。 必须 存在 具有 正确 权限 的 临时 存储 目录 。 

。 必须 存在 具有 正确 权限 的 最 终 存储 目录 。 

这 些 条 件 都 满足 后 , 方 可 确保 文件 上 传 功能 正常 工作 。 

1. 允许 文件 上 传 

正确 设置 PHP 才能 处 理 文件 上 传 , 在 php. ini 文件 中 有 多 个 设置 ,它们 规定 了 PHP 
如 何 处 理 上 传 ,包括 可 以 上 传 多 大 的 文件 ,以 及 应 该 把 上 传 的 文件 临时 存储 在 什么 位 置 。 
一 般 来 说 ,不 需 额 外 设置 即 可 实现 上 传 ,但 是 如 果 需 要 上 传 非常 大 的 文件 (超过 2MB) ,就 
必须 要 修改 php. ini。 

在 php. ini 文件 的 File Uploads 区 域 中 ,如 图 7-5 所 示 , 具 有 四 个 能 够 控制 PHP 如 何 
处 理 文件 上 传 的 指令 。 这 些 指令 和 默认 值 及 其 相关 描述 如 表 7-1 所 示 。 


S12 
S13 
S14 
515 
516 ; Whether to allow HTTP file uploads. 

S17 file_ uploads = On 

S18 

S19 ; Temporary directory for HTTP uploaded files (will use system default if not 
S20 ; specified). 

S21 ;upload tmp_dir = 

S22 

S23 ; Naximum allowed size for uploaded files. 

S24 upload max_filesize = 2 了 


7-5 ”php. ini 设置 会 影响 是 否 能 够 管理 PHP 中 的 文件 上 传 


根据 需求 设置 php. ini 文件 后 ,需要 重新 启动 Web 服务 器 使 之 生效 。 同 时 考虑 到 大 
文件 上 传 容 易 造 成 网 络 超时 的 情况 ,还 可 能 需要 在 php. ini 文件 中 更 改 max_execution_ 
time 的 值 或 者 在 脚本 中 使 用 set_time_limit( $ TimeLimit) 来 加 大 超时 限制 时 间 。 
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表 7-1 php. ini 中 关于 文件 上 传 的 设置 指令 


指 令 描 述 默认 值 
file_uploads 控制 是 否 允 许 HTTP 方式 的 文件 上 传 。 允 许 值 为 On 或 Off ON( 启 用 ) 
指定 上 传 的 文件 在 被 处 理 之 前 的 临时 保存 目录 。 如 果 没 有 设置 

upload_tmp_dir NULL 


该 选项 ,将 使 用 系统 默认 值 


控制 允许 上 传 的 文件 最 大 大 小 。 如 果 文 件 大 小 大 于 该 值 ,PHP 


upload_max_filesize 将 创建 一 个 文件 大 小 为 0 的 占 位 符 文件 2M 
控制 PHP 可 以 接受 的 ,通过 POST 方法 上 传 数据 的 最 大 值 。 该 
post_max_size 值 必须 大 于 upload_max_filesize 设置 值 ,因为 它 是 所 有 POST | 8M 


数据 的 大 小 ,包括 了 任何 上 传 的 文件 


2. 保证 文件 夹 权限 的 安全 

文件 上 传 需要 两 个 文件 夹 : 临时 目录 和 最 终 目 的 文件 夹 。PHP 将 把 上 传 的 文件 先 
存储 到 临时 目录 upload_tmp_dir 中 ,直到 PHP 脚本 把 它 移动 到 最 终 目 的 地 址 为 止 。 因 
此 ,对 于 这 两 个 文件 夹 ,Web 用 户 必须 具有 写 至 该 文件 夹 的 权限 。 

如 果 服 务 器 是 Windows 的 系统 ,那么 默认 的 系统 临时 文件 夹 为 C:\WINDOWS\ 
TEMP\, 如果 服 务 器 是 类 UNIX 系统 ,那么 默认 的 系统 临时 文件 夹 为 /tmp。 不 管 是 
Windows 还 是 UNIX, 系 统 临 时 文件 夹 的 权限 都 已 开放 ,对 于 Web 用 户 来 说 是 具有 可 写 
权限 的 。 

上 传 的 最 终 目的 文件 夹 是 由 用 户 定义 的 ,我 们 可 以 在 这 个 文件 夹 建立 在 站 点 目录 中 ， 
例如 ,uploads 文件 夹 , 并 为 该 文件 夹 设 定 允 许 每 个 人 可 读 可 写 的 权限 。 

需要 注意 的 是 ,在 uploads 文件 夹 上 设置 的 权限 一 一 每 个 人 可 读 可 写 , 是 相当 不 安全 
的 。 确 切 地 讲 ,任何 人 现在 都 可 以 移动 .复制 或 写 文件 到 uploads 文件 夹 中 (假定 他 们 知 
道 它 存在 )。 恶 意 用 户 可 以 写 一 个 PHP 脚本 到 uploads 目录 中 ,然后 在 Web 浏览 器 中 运 
行 这 个 脚本 ,从 事 破坏 活动 。 

所 以 ,出 于 安全 考虑 ,首要 选择 是 把 上 传 的 文件 存储 在 Web 目录 之 外 ,这 样 做 将 拒绝 
用 户 直接 访问 文件 ,并 避免 把 具有 宽松 权限 的 文件 夹 放 在 公开 可 访问 的 位 置 。 但 是 这 样 
做 也 可 能 会 引起 上 传 文件 无 法 访问 的 问题 。 

如 果 必 须 保持 uploads 文件 夹 公开 可 访问 ,就 需要 调整 权限 。 出 于 安全 考虑 ,理想 情 
况 下 只 希望 允许 Web 服务 器 用 户 读 、 写 和 浏览 这 个 目录 。 如 果 使 用 的 Web 服务 器 是 
Apache, 还 可 以 使 用 一 个 . htaccess 文件 来 限制 对 uploads 文件 夹 的 访问 。 例 如 ,可 以 声 
明 该 文件 夹 中 只 有 图 片 文件 是 公开 可 查看 的 ,这 意味 着 即使 把 PHP 脚本 放 在 那里 , 它 也 
不 能 执行 。 可 以 在 线 查 找 关 于 如 何 使 用 . htaccess 文件 的 信息 。 

当然 ,提高 安全 性 通常 会 降低 方便 性 。 对 于 本 章 的 示例 ,实际 上 没有 关注 安全 性 ,而 
只 考虑 了 能 够 轻松 演示 文件 上 传 过 程 的 方便 性 。 而 对 于 开发 者 来 说 ,需要 知道 潜在 的 风 
险 ,并 最 大 限度 地 减少 风险 。 


7.2.2 $_FILES 数组 结构 
PHP 是 通过 Web 表单 中 的 FILE 组 件 来 实现 文件 上 传 的 。PHP 的 文件 上 传 主要 是 
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利用 form 表单 提交 一 个 FILE 对 象 给 服务 器 。 其 中 FILE 对 象 必须 包含 Multipart/ 
form-data 的 entype 属性 。 
在 PHP 脚本 中 ,需要 处 理 的 数据 保存 在 超级 全 局 数组 $ _FILES 中 ,数组 将 具有 如 
表 7-2 所 示 的 内 容 。 
表 7-2 $_FILE 数组 元 素 


索引 含义 

name 文件 的 原始 名 称 (与 它 在 用 户 计算 机 上 的 名 称 一 样 ) 

type 文件 的 MIME 类 型 ,就 像 浏 览 器 所 提供 的 那样 

size 上 传 文件 的 大 小 ,以 字 节 为 单位 

tmp_name 在 服务 器 上 存储 上 传 文件 是 它 的 临时 路 径 , 包 含 临 时 文件 夹 与 文件 名 
error 在 上 传 过 程 中 任何 问题 相关 的 错误 代码 


文件 上 传 过 程 中 ,PHP 将 随 文件 信息 数组 一 起 返回 一 个 对 应 的 错误 代码 。 如 果 上 传 
失败 ,可 以 通过 检查 错误 代码 的 值 来 推断 错误 发 生 的 原因 。 该 代码 可 以 在 文件 上 传 时 生 
成 的 文件 数组 中 的 error 字段 中 被 找到 ,例如 ,$ _FILES[L'userfile']['error]。 错 误 代码 为 
0 一 4.6 和 7, 注 意 没 有 代码 5。 它 的 详细 情况 如 表 7-3 所 示 。 


表 7-3 $_FILEL'error] 错 误 代码 含义 


错误 代码 含义 
0 没有 错误 发 生 , 文 件 上 传 成 功 
1 上 传 的 文件 超过 了 php. ini 中 upload_max_filesize 选项 限制 的 值 
2 上 传 文件 的 大 小 超过 了 HTML 表单 中 MAX_FILE_SIZE 选项 指定 的 值 
3 文件 只 有 部 分 被 上 传 
4 没有 文件 被 上 传 
6 找 不 到 临时 文件 夹 
党 文件 写 人 失败 


如 果 已 经 知道 上 传 文件 的 位 置 及 其 名 称 , 现 在 .就 可 以 将 其 复制 到 其 他 有 用 的 地 方 。 
在 脚本 执行 结束 前 ,这 个 临时 文件 将 被 删除 。 因 此 ,如 果 要 保留 上 传 文件 ,必须 将 其 重 命 
名 或 移动 。 一 旦 PHP 脚本 接受 了 文件 ,move_uploaded_file() 函 数 就 可 以 把 它 从 临时 目 
录 传 输 到 最 终 目 的 位 置 。 


move_uploaded file ("临时 目录 "," 目 的 位 置 "); 
如 果 移 动 成 功 ,那么 就 会 把 该 文件 的 临时 版 本 移动 到 其 新 目的 地 。 


7.2.3 ”上传 综合 范例 
在 这 个 范例 中 ,将 让 用 户 选 择 他 们 本 地 计算 机 上 的 一 个 文件 ,将 其 上 传 存储 在 站 点 目 
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录 的 uploads 文件 中 ,同时 脚本 将 检查 文件 大 小 及 类 型 是 否 符合 要 求 , 要 求 文件 大 小 不 超 
过 512kb 有 目 必须 是 图 片 。 

既然 服务 器 已 经 设置 成 适当 地 允许 文件 上 传 ,就 可 以 创建 执行 实际 文件 处 理 的 PHP 
脚本 。 脚 本 分 为 两 个 部 分 : HTML 表单 和 PHP 代码 。 

1. 文件 上 传 的 HTML 表单 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 HTML 文档 (参见 脚本 7-5) 。 


脚本 7-5 创建 上 传 表单 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN” 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtml1- transitional .dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 


和 

3 

4 

5 <meta http-equiv= "Content- TYPe”content= "text/html; charset=gb2312”/> 

6 ”<title> 上 传 图片 < /title> 

7 </head> 

8 <body> 

娄 

10 <form enctype= "multipart/form- data" action= "upload image.php" method= 
"post"> 

11 <input type= "hidden" name= "MAX FILE SIZE" value= "524288" /> 

12 <fieldset><legend> 请 选择 一 张 JPEG、PNG 或 GIF 格式 的 图 片上 传 :< /legend> 

13 <P><b> 头 像 :</b><input type= "file" name= "upload" /></p> 

14 </fieldset> 

15 <div align="center">< input type="submit" name= "submit" value=" 上 传 " /> 

16 </div> 

17 </form 

18 </body> 

19 </html> 


(2) 这 个 表单 非常 简单 ,但 是 它 包含 上 传 文件 所 需 的 3 个 部 分 : 
。 开头 表单 标签 的 enctype 部 分 指示 表单 应 该 能 够 处 理 多 种 类 型 的 数据 ,包括 文件 。 
注意 ,表单 必须 使 用 post 方法 提交 。 
MAX_FILE_SIZE 隐藏 字段 用 于 限制 所 选 的 文件 可 以 为 多 大 (以 字 节 为 单位 ) ,并 
且 必须 在 文件 输入 框 之 前 。 
file 输入 类 型 将 在 表单 中 创建 文件 浏览 按钮 。 

(3) 创建 提交 按钮 ,并 结束 表单 ,完成 页 面 。 将 文件 另存 为 upload_form. html, 并 在 
Web 浏览 器 中 测试 它们 (参见 图 7-6) 。 

特别 注意 ,遗漏 了 enctype 表单 属性 是 文件 上 传 失 败 的 常见 理由 。MAX_FILE_ 
SIZE 是 浏览 器 中 对 于 文件 大 小 的 一 种 限制 .尽管 并 非 所 有 的 浏览 器 都 会 遵守 这 个 限制 。 
PHP 配置 文件 有 它 自己 的 限制 。 
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Er 
请 选择 一 张 JPEG 、PNG 或 GIF 格式 的 图 片上 传 : 
头像 : CE. 


[E33 


图 7-6 上 传 文件 的 HTML 表单 


2. 编写 处 理 文件 的 PHP 
编写 获取 上 传 文件 的 PHP 代码 是 相当 直观 和 简单 的 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 .从 HTML 开始 (参见 脚本 7-6) 。 


脚本 7-6 人 允许 用 户 上 传 一 个 图 像 到 服务 器 ,并 正确 显示 它 。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 

<html xmlns= "http://www.w3.0org/1999/xhtml"> 

<head> 

<meta http- equiv= "Content- Type" content= "text/html; charset=gb2312" /> 
<title> 上 传 图 片 < /title> 

< /head> 

<body> 


< ?php #Script 7- 6 - upload image.php 
if (isset ($_FILES["'upload'])){ 
if($_FILES["'upload'] ['error']==2){ 
echo '<p><font color= "red"> 请 上 传 一 个 小 于 512K 的 图 片 。< /font>< /p> '; 
}else{ 
$allowed=array ('image/jpg', 'image/jpeg', 'image/png', 'image/pjpeg’', 
"image/gif', 'image/x-png'); 
if(in array($_FILES["'upload'] ['type'], $allowed)){ 
if (move_uploaded file($_FILES["'upload'] ['trp_name'], "uploads/ {$_FILES[' 
Wpload'] ['name']}")){ 
echo '<p> 文 件 已 被 上 传 。< /p> '; 
echo "< img src= \"uploads/{$_FILES['upload'] ['name']}\" />"; 
} else { 
echo '<p>< font color="red"> 文 件 上 传 有 误 : < /b>'; 
switch($_FILES['upload'] ['error']){ 
case 1: 


echo ' 文 件 超出 php.ini 中 upload max filesize 指定 大 小 。'; 


153 


154 


PHP+MySQL 项 目 实例 开发 


25 break; 

26 Case 2: 

27 echo ' 文 件 超出 表单 中 设 定 的 MAX_FILE SITZE 的 指定 大 小 。'; 
28 break; 

29 Case 3: 

30 echo ' 文 件 仅 被 部 分 上 传 。'; 

31 break; 

3 Case 4: 

33 echo "文件 未 被 上 传 。'; 

34 break; 

35 Case 6: 

36 echo ' 没 有 可 用 的 临时 文件 夹 。'; 
37 break; 

38 default: 

39 echo ' 系 统 错误 。'; 

40 break; 

41 } 

42 echo '</b>< /font>< /p>"'; 

43 } 

44 } else { 

45 echo '<p><font color="red"> 请 选择 图 片 格式 文件 上 传 。< /font>< /p>'; 
46 unlink($_FILES["upload'] ['tmp_name']); 

47 } 

48 } 

49 } 

50 ?> 

51 

52 </body> 

53 </html> 


(2) 检查 表单 提交 后 ,是 否 正确 选择 一 个 文件 。 
if (isset($_FILES["'upload'])){ 


因为 这 个 表单 没有 其 他 的 字段 要 验证 ,这 是 所 需 的 唯一 一 个 条 件 语 句 。 
(3) 验证 是 否 超出 浏览 器 的 MAX_FILE_SIZE 大 小 。 


if($_FILES["'upload'] ['error']==2){ 


请 特别 注意 这 一 个 条 件 语 句 , 当 设 定 浏览 器 的 MAX_FILE_SIZE 小 于 php. ini 中 
upload_max_filesize 的 值 ,并 且 上 传 文件 大 于 MAX_FILE_SIZE 的 值 ,但 小 于 PHP 配置 
文件 php. ini 中 upload_max_filesize 指定 大 小 的 值 时 ,也 会 创建 $ _FILE 数组 ,但 不 会 进 
行 上 传 ,$ _FILE 数组 的 [hame"] 元 素 将 存放 上 传 文件 名 称 , $ _FILE 数组 的 [errorq] 元 素 
的 值 将 会 为 2, 提 示 错 误 ,而 其 他 元 素 都 为 空 。 所 以 必须 先 做 这 个 判断 ,然后 再 判断 文件 
名 等 。 但 若是 MAX_FILE_SIZE 大 于 php. ini 中 upload_max_filesize 的 值 , 则 无 须 执行 
这 一 条 件 判 断 。 
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(4) 检查 上 传 的 文件 是 正确 的 类 型 。 

$allowed= array (' image/jpg ', ' image/jpeg ', ' image/png ', ' image/pjpeg ', ' image/gif ', ， 

image/x- png'); 

if(in array($_FILES['upload'] ['type'], $allowed)){ 

文件 的 类 型 是 MIME 类 型 ,指示 它 是 哪 一 类 文件 ,浏览 器 将 依赖 于 所 需 文件 的 属性 
来 确定 和 提供 这 种 信息 。 一 般 来 说 ,大 多 数 浏览 器 支持 的 图 像 通常 具有 image/jpg、 
image/jpeg、image/png、image/pjpeg、image/gif 及 image/x-png 类 型 。 为 了 验证 文件 类 
型 ,首先 创建 一 个 允许 类 型 的 数组 。 如 果 上 传 文件 的 类 型 在 这 个 数组 中 ,文件 就 是 有 效 
的 ,并 且 应 该 进行 处 理 。 

(5) 把 文件 复制 到 服务 器 上 的 新 位 置 。 


if (move_ uploaded file($_FILES['upload']['tmp name'], "uploads /{$_FILES 

["'upload'] ['name']}")){ 

使 用 move_uploaded_file() 函数 把 临时 文件 移动 到 它 的 目的 位 置 , 即 uploads 文件 夹 
中 ,该 文件 将 保留 其 原始 名 称 。 若 成 功 移动 ,该 函数 将 会 返回 True, 所 以 应 该 总 是 使 用 一 
个 条 件 语句 来 确认 ,而 不 是 仅仅 假设 已 移动 。 

(6) 显示 成 功 信息 ,并 显示 刚 上 传 的 图 片 。 


echo '<p> 文 件 已 被 上 传 。</p> '; 
echo "< img src=\"uploads/{$_FILES['upload'] ["name"]}\”/> "7 


因为 已 知 上 传 文件 的 最 终 位 置 ,所 以 可 以 直接 使 用 HTML 的 二 img 之 标签 来 显示 图 
片 并 指定 图 片 的 URL。 
(7) 如 果 文件 不 能 被 移动 , 则 报告 错误 。 


switch($_FILES['upload'] ['error']){ 

case 1: 
echo ' 这 个 文件 超出 php.ini 中 设 定 的 upload max_ filesize 指定 大 小 。'; 
break; 

Case 2: 
echo ' 这 个 文件 超出 表单 中 设 定 的 MAX_FILE_sIZE 的 指定 大 小 。'; 
break; 

Case 3: 
echo ' 文 件 仅 被 部 分 上 传 。'; 
break; 

case 4: 
echo ' 文 件 未 被 上 传 。'; 
break; 

Case 6: 
echo ' 没 有 可 用 的 临时 文件 夹 。'; 
break; 

default: 
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echo ' 系 统 错误 。'; 
break; 
} 


有 多 种 可 能 的 原因 使 得 不 能 移动 一 个 文件 。 这 里 的 switch 条 件 语句 根据 错误 编号 
打印 出 问题 。 由 于 $ _FILES['upload']['error"] 变 量 可 能 并 非 总 是 具有 一 个 值 ,所 以 增加 
了 一 种 默认 情况 。 

(8) 完成 条 件 语 句 和 PHP 部 分 。 

}else{ 


echo '<p><font color= "red"> 请 选择 图 片 格式 文件 上 传 。< /font>< /p>'; 
unlink($_FILES['upload'] ['tmp_ name']); 


} 

2> 

这 个 else 语句 结束 类 型 in_array() 条 件 语句 。 如 果 文 件 不 具有 正确 的 类 型 , 则 打印 
一 条 出 错 信息 。 此 外 ,还 会 使 用 unlink() 函 数 从 服务 器 中 删除 上 传 在 临时 目录 中 的 文件 。 

(9) 完成 HTML 页 面 , 将 文件 另存 为 upload_image. php, 上 传 到 Web 服务 器 上 与 
upload_form. html 相同 的 目录 中 ,并 在 Web 浏览 器 中 测试 这 两 个 文档 (参见 图 7-7、 图 7-8 和 
图 7-9)。 


请 上 传 一 个 小 于 512k 的 图 片 。 


图 7-7 上 传 文件 超出 MAX_FILE_SIZE 的 大 小 ,提示 错误 


“I lcd 
[Bolero p- cxl| 瑟 tea | 
请 选择 图 片 格式 文件 上 传 。 有 


图 7-8 上 传 了 无 效 的 文件 类 型 ,提示 错误 


另外 ,可 以 利用 is_uploaded_file() 两 数 来 验证 上 传 文件 是 否 存在 ,还 可 以 利用 mime 
_content_type() 函数 来 检测 文件 的 MIME 类 型 。 如 果 新 上 传 的 文件 与 原 有 的 文件 具有 
相同 的 名 称 ,那么 move_uploaded_file() 函 数 将 覆盖 原 有 的 文件 ,而 不 会 发 出 警告 。 


第 7 章 ”创建 动态 Web 站 点 


全 htpyV/localhosybc 只 ~- OX 


图 7-9 上 传 成 功 将 显示 该 图 片 


7.3 项 目 训练 简易 计数 器 设计 


7.3.1 项 目 说 明 


网 站 计数 器 是 Web 应 用 程序 的 一 项 基本 功能 ,用 于 统计 使 用 网 站 或 者 应 用 程序 的 人 
数 ,可 反映 出 网 站 或 者 应 用 程序 的 受 欢迎 程度 ,对 于 网 站 可 信 度 的 研究 有 一 定 的 影响 。 本 
节 将 为 网 站 创建 一 个 访问 者 计数 器 。 

网 页 看 上 去 组 织 得 很 好 并 显示 了 图 像 和 文本 ,但 具体 的 底层 结构 实际 上 是 由 大 量 文 
件 组 成 的 ,这 些 文件 存储 在 有 清晰 结构 的 目录 中 。 网 站 文件 结构 通常 由 在 一 个 启用 了 服 
务 器 的 计算 机 上 运行 并 具有 特定 格式 的 文件 组 成 。 

可 以 使 用 PHP 来 查看 当前 计算 机 的 内 部 结构 。PHP 不 仅 可 以 有 效 地 查看 和 浏览 给 
定 服 务 器 的 文件 结构 ,在 管理 文件 结构 方面 也 不 逊色 。 巾 于 PHP 能 完成 文件 的 创建 、 导 
航 、 删 除 和 修改 ,所 以 它 是 动态 维护 Web 目录 的 一 个 强大 工具 。PHP 在 打开 、 关 闭 、 读 、 
写 和 处 理 文本 文件 方面 能 力 很 强 , 包 括 HTML JavaScript 和 XML 文件 。 


7.3.2 设计 原理 


1. 打开 文件 

使 用 一 个 文件 时 (不 论 是 . txt 文件 、. xml 文件 ,还 是 其 他 文件 ) ,首先 必须 先 打开 它 以 
便 读 取 或 写 人 ,或 者 这 两 种 任务 都 执行 。PHP 提供 了 fopen() 函 数 来 做 到 这 一 点 。 

fopen() 函 数 语 法 如 下 : 


fopen (filename,mode); 


参数 filename 是 必需 的 .指定 了 要 打开 的 文件 或 URL。 
参数 mode 是 必需 ,指定 要 求 到 该 文件 / 流 的 访问 类 型 。 表 7-4 显示 了 打开 一 个 文件 
的 多 种 其 他 方法 。 如 果 没 有 向 fopen() 函数 提供 mode 参数 来 指定 如 何 打开 文件 ,PHP 
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会 发 出 一 个 警告 ,文件 将 不 能 正确 打开 。 


表 7-4 fopen 函数 mode 参数 选项 


mode 


说 明 


只 读 方式 打开 ,将 文件 指针 指向 文件 头 


读 写 方式 打开 ,将 文件 指针 指向 文件 头 


写 人 方式 打开 ,将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文件 不 存在 , 则 尝试 创 
建 之 


读 写 方式 打开 ,将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文件 不 存在 , 则 尝试 创 
建 之 


写 人 方式 打开 ,将 文件 指针 指向 文件 末尾 。 如 果 文 件 不 存在 , 则 尝试 创建 之 


读 写 方式 打开 ,将 文件 指针 指向 文件 末尾 。 如 果 文 件 不 存在 , 则 尝试 创建 之 


创建 并 以 写 人 方式 打开 ,将 文件 指针 指向 文件 头 。 如 果 文件 已 存在 , 则 fopen() 调 用 失败 
并 返回 FALSE, 并 生成 一 条 E_WARNING 级 别 的 错误 信息 。 如 果 文 件 不 存在 , 则 尝试 创 
建 之 


创建 并 以 读 写 方式 打开 ,将 文件 指针 指向 文件 头 。 如 果 文件 已 存在 , 则 fopen() 调 用 失败 
并 返回 FALSE, 并 生成 一 条 E_WARNING 级 别 的 错误 信息 。 如 果 文件 不 存在 , 则 尝试 创 
建 之 


fopen() 函 数 返回 一 个 文件 源 , 我 们 稍 后 可 以 使 用 它 来 操作 打开 的 文件 。 
例如 要 打开 一 个 文件 以 供 读 取 : 


$ fp= fopen ("test.txt", "r"); 


要 打开 一 个 文件 以 便 写 入 : 


$ fp= fopen ("test .txt", "w"); 


要 打开 一 个 文件 以 便 添加 内 容 , 即 在 文件 末尾 添加 数据 : 


$ fp= fopen ("test.txt", "a"); 


如 果 由 于 许多 原因 而 不 能 打开 文件 ,fopen() 函 数 返回 false。 因 此 ,在 使 用 文件 之 前 
测试 函数 的 返回 值 ,这 是 个 不 错 的 方法 。 我 们 可 以 使 用 一 条 让 语句 来 做 到 这 一 点 : 


if ($fp=fopen("test.txt", "w")){ 


} 


// 处 理 文件 


或 者 ,如 果 一 个 基本 文件 无 法 打开 , 则 可 以 使 用 一 个 逻辑 操作 符 来 结束 执行 : 


($ fp= fopen ("test.txt","w"))or die ("对 不 起 ,无 法 打开 文件 。"); 

如 果 fopen( 〇 函数 返回 true, 表 达 式 的 剩 下 部 分 将 不 会 解析 ,并 且 die() 函 数 (该 函数 
会 向 浏览 器 写 人 一 条 消息 并 结束 脚本 ) 不 会 执行 。 否 则 .or 操作 符 的 右 表达 式 将 会 解析 ， 
而 且 调 用 die() 函 数 。 
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很 自然 地 ,作为 一 个 深思 熟 虑 的 开发 人 员 ,应 该 认识 到 可 能 会 出 现 异常 ,因此 必须 验 
证 异常 情况 以 免 出 现 意 外 。 应 当 记 住 ,要 打开 的 文件 必须 有 适当 的 权限 许可 ,从 而 允许 读 
或 写 , 因 此 ,如 果 使 用 的 是 Linux 服务 器 ,就 应 该 使 用 chmod 命令 为 文件 授权 ,允许 读 文 
件 。 如 果 使 用 Windows 服务 器 , 则 应 当 检 查 读 / 写 权限 设置 。 一 般 来 说 ,如 果 不 是 特别 关 
心 一 个 特定 文件 的 安全 性 ,应 把 它 设置 为 777。 不 过 ,如 果 很 关注 某 个 给 定 可 写 文件 的 安 
全 性 , 则 应 当 考虑 一 种 更 高 级 的 方法 来 保护 这 个 文件 (如 htaccess)。 

2. 读 文件 

PHP 为 读 文 件 提供 了 很 好 的 方法 。 不 仅 可 以 打开 文件 来 读数 据 , 还 可 以 由 PHP 创 
建文 件 并 打开 这 个 文件 。 

从 文本 文件 获取 信息 时 ,最 常见 的 方法 是 使 用 函数 fgetc() \fgets() 和 fread()。 这 些 
函数 的 用 法 很 类 似 , 不 过 各 有 优 缺 点 ,关键 是 要 用 其 所 长 。 在 这 里 详细 介绍 最 常见 的 
fread() 函 数 ,语法 如 下 : 


fread (file, length); 


参数 file 是 必需 的 ,指定 要 读 取 打 开 文 件 ,参数 length 也 是 必需 的 ,规定 要 读 取 的 最 
大 字 节 数 。 

fread() 从 文件 指针 file 读 取 最 多 length 个 字 节 。 该 函数 在 读 取 完 最 多 length 个 字 
节 数 ,或 到 达 文件 尾 时 就 会 停止 读 取 文件 ,取决 于 先 碰 到 哪 种 情况 。 

由 于 我 们 不 知道 访问 的 文件 会 有 多 大 ,所 以 可 以 结合 使 用 freadO) 函 数 和 filesize() 函 
数 (filesize() 函 数 会 返回 当前 文件 的 大 小 ) ,完整 地 读 出 当前 文件 的 内 容 , 从 而 考虑 到 所 有 
可 能 性 。 例 如 : 

$ file=fopen("test.txt", "r"); 

fread ($ file, filesize ("test.txt")); 

3. 写 文件 

使 用 文件 时 ,还 需要 知道 如 何 改变 文件 :也 就 是 说 ,要 知道 如 何 写 文件 。 类 似 于 读 文 
件 ,可 以 采用 多 种 方法 写 文件 。 完 成 这 个 任务 最 常用 的 方法 是 使 用 fwrite() 函数, 语法 
如 下 : 

fwrite (file, string, length); 

参数 file 是 必需 的 ,指定 了 要 写 和 人 的 打开 文件 。 参 数 string 指定 了 要 写 人 文件 的 字 
符 串 。 参 数 length 是 可 选 的 ,指定 了 要 写 人 的 最 大 字 节 数 。 

fwrite() 把 string 的 内 容 写 入 文件 指针 file 处 。 如 果 指 定 了 length, 当 写 入 了 length 
个 字 节 或 者 写 完 了 string 以 后 , 写 人 就 会 停止 ,取决 于 先 碰 到 哪 种 情况 。fwrite() 返回 写 
入 的 字符 数 , 出 现 错误 时 则 返回 false。 

4. 关闭 文件 

一 且 使 用 完 文 件 ,最 后 要 做 的 就 是 完成 清理 工作 。 所 谓 清 理 , 是 指 必 须 关 闭 当 前 的 文 
件 指针 。 文 件 的 关闭 操作 可 以 释放 与 文件 相关 的 资源 ,而 且 避 免 之 后 代码 中 因为 重复 利 
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用 文件 句柄 变量 而 可 能 导致 的 冲突 。 
PHP 中 关闭 文件 与 打开 文件 同样 简单 ,只 需 调 用 fclose() 方 法 , 它 会 关闭 文件 指针 链 
接 。fclose() 函 数 语 法 如 下 : 


fclose (file); 


参数 file 指定 了 要 关闭 的 文件 ,file 参数 是 一 个 文件 指针 。fclose() 函 数 关 闭 该 指针 
指向 的 文件 。 如 果 成 功 则 返回 true, 否则 返回 false。 文 件 指针 必须 有 效 , 并 且 是 通过 
fopen() 成 功 打开 的 。 

例如 : 


$file= fopen ("test.txt"v"r") 7 
// 执 行 的 一 些 代码 … 


fclose($ file); 


7.3.3 设计 过 程 


下 面 来 为 网 站 创建 一 个 访问 者 计数 器 ,首先 需要 创建 一 个 文件 用 于 保存 当前 的 访问 
量 ,这 个 文件 可 以 是 一 个 文本 文件 ,例如 ,times. txt。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 .从 HTML 开始 (参见 脚本 7-7) 。 


脚本 7-7 访问 文件 ,并 将 计数 器 加 1。 


< !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.0rg/TR/xhtml1/DTD/xhtmll- transitional.dtd"> 
<html xmlns= "http://www.w3.org/1999/xhtml"> 

<head> 


1 
2 
EE 
4 
5 <meta http-equiv= "Content- Type" content="text/html; charset=gb2312" /> 
6 ”<title> 简 易 计 数 器 < /title> 

7 </head> 

8 <body> 

3 

10 <?php #5Script 7-8-counter.php 

11 $fp=fopen("times.txt", "r"); 

12 $str=fread($ fp, filesize("times.txt")); 

13 $strt+? 

14 fclosel($ fp); 

15 $fp=fopen("times.txt", "w"); 

16 fwrite($ fp, $str); 

17 fclose($ fp); 

18. 2> 


20 您 是 本 站 的 第 <?Php echo $ str?> 个 访问 者 。 
21 </body> 
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22 </html> 

(2) 以 读 形式 打开 记录 以 往 访问 人 数 的 文件 times. txt。 

$ fp= fopen ("times .txt", "r"); 

(3) 从 文件 中 读 人 数值 。 

$str=fread($ fp, filesize ("times.txt")); 

由 于 我 们 不 知道 计数 器 会 有 多 大 ,所 以 需 结合 使 用 fread() 函 数 和 filesize() 函数 
(filesize( ) 函数 会 返回 当前 文件 的 大 小 ) 完 整地 读 出 当前 计数 器 ,从 而 考虑 到 所 有 可 能 性 。 
使 用 fread() 函数 读 入 了 计数 器 的 当前 值 ,现在 将 变量 $ curvalue 赋 为 计数 器 的 当前 值 。 

(4) 计数 器 加 1。 

$str+t+? 

(5) 关闭 文件 。 

fclose($ fp); 

(6) 重新 以 写 的 方式 打开 记录 访问 人 数 的 文件 times. txt。 

$ fp= fopen ("times.txt", "w"); 

(7) 把 最 新 的 访问 人 数 写 入 文件 。 

fwrite($ fp, $str); 

(8) 再 次 关闭 文件 。 

fclose($ fp); 


通过 关闭 文件 指针 链接 作为 读 写 操 作 的 最 终 工作 。 如 果 做 到 了 这 一 点 ,就 能 正确 地 
输出 当前 计数 。 

(9) 完成 PHP 代码 块 ,输出 计数 器 数值 。 

您 是 本 站 的 第 二 ? php echo $ str? 二 个 访问 者 。 

(10) 完成 HTML 页 面 ,将 文件 另存 为 counter. php, 上 传 到 Web 服务 器 ,并 在 Web 
浏览 器 中 测试 这 个 文档 ,如 图 7-10 所 示 。 


铭 htpy/localhosybec 只 - EX 


您 是 本 站 的 第 85 个 访问 者 。 


图 7-10 从 文件 中 读 取 计数 器 
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上 述 代码 完成 一 个 简易 的 计数 器 ,每 次 刷新 该 页 面 ,计数 器 会 自动 加 一 。 

在 文件 操作 过 程 中 ,需要 注意 的 是 ,如 果 一 个 文件 无 法 打开 以 便 写 入 或 读 取 ,我 们 应 
该 结束 脚本 执行 。 如 果 脚 本 绝对 依靠 我 们 想 要 使 用 的 文件 ,应 该 使 用 dieO 〇 函数, 向 浏览 
器 显示 一 条 含有 信息 的 错误 消息 。 在 重要 程度 较 低 的 情况 下 ,仍然 需要 考虑 到 失效 ,可 能 
会 考虑 将 其 写 人 到 一 个 日 志文 件 中 。 


在 本 章 中 ,学 习 了 如 何 使 用 include() 等 函数 (include_once()、require() 和 require_ 
once()) 来 把 文件 包含 到 文档 中 ,并 且 执 行 包 含 文件 中 的 任何 PHP 代码 。 学 习 了 如 何 读 
取 文 件 . 写 入 到 文件 ,并 完成 了 一 个 图 像 上 传 的 范例 。 


重点 回顾 
1. 四 个 包含 函数 的 用 法 。 
2. PHP 上 传 文件 方法 。 
3. 文件 读 写 函数 的 应 用 。 

本 章 实 训 


【 实 训 1】 

修改 第 6 章 实 训 3 的 nn. php, 将 写成 函数 的 算法 另外 保持 至 外 部 包含 文件 ,通过 包 
含 函 数 将 代码 包含 进来 ,页 面 功能 依然 是 使 用 文本 框 输入 的 变量 传递 参数 ,验证 是 否 输 入 
为 数字 (提示 : 使 用 intval 函数 ) ,验证 正确 后 将 结果 输出 为 表格 格式 。 

【 实 训 2】 

新 建 counterimg. php 页 面 ,结合 脚本 5-8 和 脚本 7-8, 实 现 计数 器 ,并 将 数字 替换 为 
图 片 显示 。 

【 实 训 3】 

修改 第 6 章 实 训 4 submit. php, 添 加 上 传 头像 的 功能 .要 求 单 击 “ 注 册 ” 按 钮 后 ,显示 
用 户 的 输入 ,并 显示 头像 图 片 。 网 页 效果 如 图 7-11 所 示 。 
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您 的 输入 信息 为 


基本 信息 ( * 为 必 填 项 ) 

用 户 和 名 * 

密码 * 123456 
mary1638126.com 


艾 条 出 生 的 城市 
上 海 


女 

1985 年 11 月 1! 日 

北京 

http://mrr. naryblog, com” 
80833333 

mary1638asn_ com 


音乐 电影 运动 


Nello world! 


图 7-11 submit. php 的 运行 效果 
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8.1 数据 库 设 计 


良好 的 数据 库 设 计 对 于 一 个 高 性 能 的 应 用 程序 非常 重要 ,就 像 一 个 空气 动力 装置 对 
于 一 辆 赛车 的 重要 性 一 样 。 如 果 一 辆 汽车 没有 平滑 的 曲线 ,在 行驶 时 将 会 产生 阻力 从 而 
变 慢 。 关 系 没 有 经 过 优化 ,数据 库 无 法 尽 可 能 高 效 地 运行 。 应 该 把 数据 库 的 关系 和 性 能 
看 作 是 规范 化 的 一 部 分 。 

一 个 规划 和 设计 良好 的 数据 库 的 优点 是 众多 的 , 它 也 证 实 了 这 样 一 个 道理 : 前 期 做 
的 工作 越 多 ,后 面 所 要 做 的 就 越 少 。 在 使 用 数据 库 的 应 用 程序 公开 发 布 之 后 ,还 要 对 数据 
库 进行 重新 设计 ,这 是 最 糟糕 的 ,然而 这 确实 会 发 生 , 并 且 代 价 高 郧 。 


8.1.1 收集 并 分 析 数 据 需求 


需求 收集 和 分 析 是 数据 库 设 计 的 第 一 阶段 ,这 一 阶段 收集 到 的 基础 数据 和 一 组 数据 
流 图 是 下 一 步 设计 概念 结构 的 基础 。 概 念 结构 是 整个 组 织 中 所 有 用 户 关心 的 信息 结构 ， 
对 整个 数据 库 设计 具有 深刻 影响 。 而 要 设计 好 概念 结构 ,就 必须 在 需求 分 析 阶 段 用 系统 
的 观点 来 考虑 问题 ,收集 和 分 析 数 据 及 其 处 理 方法 或 流程 。 

需求 分 析 阶 段 的 任务 是 通过 详细 调查 现实 世界 要 处 理 的 对 象 (组 织 、 部 门 、 企 业 等 )， 
充分 了 解 原 系统 的 工作 概况 ,明确 用 户 的 各 种 需求 .然后 在 此 基础 上 确定 新 系统 的 功能 。 
新 系统 必须 考虑 今后 可 能 的 扩充 和 改变 ,不 能 今 按照 当前 的 应 用 需求 来 设计 数据 库 。 

调查 的 重点 是 “数据 "和 “处 理 ”, 通 过 调查 要 从 中 获得 每 个 用 户 对 数据 库 的 如 下 要 求 : 

(1) 信息 要 求 。 指 用 户 需 要 从 数据 库 中 获得 信息 的 内 容 与 性 质 。 由 信息 要 求 可 以 得 
出 数据 要 求 , 即 在 数据 库 中 需 存储 哪些 数据 。 

(2) 处 理 要 求 。 指 用 户 要 完成 什么 处 理 功能 ,对 处 理 的 响应 时 间 有 何 要 求 , 处 理 的 方 
式 是 批 处 理 还 是 连 机 处 理 。 

(3) 安全 性 和 完整 性 的 要 求 。 

为 了 很 好 地 完成 调查 的 任务 ,设计 人 员 必 须 不 断 地 与 用 户 交 流 ,与 用 户 达 成 共识 ,以 
便 逐 步 确定 用 户 的 实际 需求 ,然后 分 析 和 表达 这 些 需求 。 需 求 分 析 是 整个 设计 活动 的 基 
础 ,也 是 最 困难 .最 花 时 间 的 一 步 。 

在 实际 开展 需求 分 析 工 作 时 有 两 点 需要 特别 注意 : 

第 一 , 在 需求 分 析 阶 段 一 个 重要 而 困难 的 任务 是 收集 将 来 应 用 所 涉及 的 数据 。 若 设 
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计 人 员 仅 仅 按 当 前 应 用 来 设计 数据 库 , 新 数据 的 加 入 不 仅 会 影响 数据 库 的 概念 结构 ,而 且 将 影 
响 逻 辑 结构 和 物理 结构 ,因此 设计 人 员 应 充分 考虑 到 可 能 的 扩充 和 改变 ,使 设计 易于 更 改 。 
第 二 , 必须 强调 用 户 的 参与 ,这 是 数据 库 应 用 系统 设计 的 特点 。 数 据 库 应 用 系统 和 
广泛 的 用 户 有 密切 的 联系 ,其 设计 和 建立 又 可 能 对 更 多 人 的 工作 环境 产生 重要 影响 。 因 
此 ,设计 人 员 应 该 和 用 户 充 分 合作 进行 设计 ,并 对 设计 工作 的 最 后 结果 承担 共同 的 责任 。 


8.1.2 逻辑 地 划分 数据 


设计 包括 通过 可 用 的 软件 工具 来 利用 在 分 析 过 程 中 所 发 现 的 知识 ,并 判断 分 析 结 果 
所 适用 的 场合 。 分 析 阶 段 决定 应 该 创建 的 事物 。 设 计 阶 段 通 过 确定 创建 表 的 方法 将 分 析 
结果 进一步 加 以 利用 。 这 一 阶段 包括 表 、 表 的 字段 和 数据 类 型 。 更 为 重要 的 是 ,设计 阶段 
包括 将 所 有 要 素 联系 在 一 起 的 方法 。 

分 析 是 对 表 、 表 中 的 信息 以 及 表 与 表 之 间 的 基本 关系 作出 定义 。 将 设计 阶段 的 所 有 
要 素 以 最 基本 的 形式 联系 在 一 起 就 是 对 参照 完整 性 的 精确 定义 。 由 于 参照 完整 性 
(Referential integrity) 是 对 表 与 表 之 间 关 系 的 加 强 , 因 此 它 是 一 种 设计 过 程 。 表 与 表 之 
间 关系 的 最 初 建立 是 分 析 过 程 ,而 不 是 设计 过 程 。 

换言之 ,分 析 定 义 要 做 什么 ;设计 则 组 织 如 何 做 。 设 计 阶 段 引入 数据 库 建 模 细 化 过 程 
的 概念 ,例如 ,规范 化 和 非 规范 化 。 根 据 应 用 程序 的 构造 ,设计 阶段 将 开始 为 前 端 用 户 进 
行 “ 零 俯 ? 地 定义 ,例如 报表 和 图 形 用 户 界面 (GUI) 的 屏幕 。 

以 图 形 化 方式 构建 表 、 添 加 字段 .定义 数据 结构 以 及 应 用 参照 完整 性 。 通 过 规范 化 和 
非 规范 化 的 过 程 可 以 达到 细 化 的 目的 。 

在 开始 使 用 SQL 和 MySQL 之 前 ,必须 确定 应 用 程序 的 需要 ,指定 数据 库 设计 。 对 
于 本 章 中 的 示例 ,将 创建 一 个 数据 库 , 命 名 为 mydatabase, 用 于 存储 一 些 用 户 注 册 信 息 。 
该 数据 库 将 包含 单独 一 张 表 users, 它 包含 用 于 存储 用 户 ID 、 姓 名、 电子 邮件 地 址 、 密 码 和 
注册 日 期 的 列 。 表 8-1 显示 了 当前 布局 ,并 且 为 列 标题 使 用 MySQL 的 命名 规则 ,即使 用 
字母 数字 以 及 下 划 线 ,不 含 空格 。 


表 8-1 users 表 将 具有 5 列 , 用 于 存储 各 种 记录 ,如 这 里 的 示例 数据 


列 名 示 例 列 名 示 例 
user_id 050 email mary@126. com 
name Mary registration_date 2014-10-10 09:00:00 
password 123456 


8.1.3 选择 正确 的 数据 类 型 


一 旦 确定 了 数据 库 所 需 的 所 有 表 和 列 ,就 应 该 确定 每 个 字段 的 MySQL 数据 类 型 。 
1. MySQL 数据 类 型 
在 创建 数据 库 时 ,MySQL 要 求 定 义 每 个 字段 将 包含 哪 一 类 信息 。 主 要 有 3 类 信息 ， 
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对 于 几乎 所 有 的 数据 库 应 用 程序 皆 是 如 此 。 


文本， 
数字 。 
。 日 期 和 时 间 。 


这 些 类 型 中 的 每 一 种 都 有 许多 可 以 使 用 的 变 体 ,其 中 有 一 些 是 MySQL 特有 的 。 正 
确 选择 列 类 型 不 仅 指 示 可 以 存储 什么 信息 ,还 会 影响 数据 库 的 总 体 性 能 。 表 8-2 列 出 了 
MySQL 的 大 多 数 可 用 类 型 .它们 占据 多 大 的 空间 以 及 每 种 类 型 的 简要 描述 。 


表 8-2 可 用 于 定义 列 的 常见 MySQL 数据 类 型 


类 型 大 小 描 述 
CHAR[Length] Length 字 节 定 长 字段 ,长 度 为 0 一 255 个 字符 
VARCHAR [Length] String 长 度 十 1 字 节 变 长 字段 ,长 度 为 0 一 255 个 字符 
TINYTEXT String 长 度 十 1 字 节 字符 串 , 最 大 长 度 为 255 个 字符 
TEXT String 长 度 十 2 字 节 字符 串 , 最 大 长 度 为 65 535 个 字符 
MEDIUMTEXT String 长 度 十 3 字 节 字符 串 , 最 大 长 度 为 16 777 215 个 字符 
LONGTEXT String 长 度 十 4 字 节 字符 串 ,最 大 长 度 为 4 294 967 295 个 字符 
TINYINT[Length] 1 字 节 范围 : 一 128 一 127, 或 者 0 一 255( 无 符号 ) 
SMALLINT[Length] | 2 字 节 es 一 32 768 一 32 767, 或 者 0 一 65 535( 无 
MEDIUMINT[Length] | 3 字 节 el 证 Cs 0 
范围 : 一 2 147 483 648 一 2 147 483 647, 或 者 
INT[Length] 4 0~4 294 967 295( 无 符号 ) 
范围 : 一 9 223 372 036 854 775 808 一 
BIGINTLLength] 8 字 节 9 223 372 036 854 775 807 ,或 者 0 一 
18 446 744 073 709 551 615( 无 符号 ) 
FLOAT 4 字 节 具有 浮动 小 数 点 的 较 小 的 数 
DOUBLE 【 Length， | 8 字 节 具有 浮动 小 数 点 的 较 大 的 数 


Decimals] 


DECIMAL [ Length，| Length 十 1 字 节 或 Length 十 | 存储 为 字符 串 的 DOUBLE, 允许 固定 的 小 

Decimals] 2 字 节 数 点 

DATE 3 字 节 采用 YYYY-MM-DD 格式 

TIME 3 字 节 采用 HH:MM:SS 格式 

DATETIME 8 字 节 采用 YYYY-MM-DD HH:MM:SS 格式 
采用 YYYYMMDDHHMMSS 格式 ;可 接受 

TIMESTAMP 4 字 节 的 范围 终止 于 2037 年 
enumeration( 枚 举 ) 的 简写 ,这 意味 着 每 一 列 

a Eas 都 可 以 具有 多 个 可 能 的 值 之 一 

SET 1.2.3.4 或 8 字 节 与 ENUM 一 样 ,只 不 过 每 一 列 都 可 以 具有 多 


个 可 能 的 值 
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许多 类 型 可 以 带 有 可 选 的 Length 属性 ,以 限制 它们 的 大 小 ,其 中 方 括 号 [指示 要 放 
置 在 圆 括号 中 的 可 选 参数 。 应 该 记 住 ,如果 把 一 个 长 度 为 5 个 字符 的 字符 串 插入 到 一 个 
CHAR(2) 字 段 中 ,就 会 截 去 最 后 3 个 字符 。 对 设置 了 其 长 度 的 任何 字段 (CHAR、 
VARCHAR INT 等 ) 都 是 如 此 。 因 此 ,长 度 应 该 总 是 对 应 于 可 以 存储 的 可 能 最 大 的 值 
(对 于 数字 ) ,或 者 可 能 最 长 的 字符 串 ( 对 于 文本 ) 。 

需要 特别 说 明 的 是 CHAR 与 VARCHAR: 这 两 种 类 型 都 存储 字符 串 , 并 且 可 以 设 
置 固定 的 最 大 长 度 。 它 们 之 间 的 一 个 主要 区 别 是 ,存储 为 CHAR 的 任何 内 容 将 总 是 被 存 
储 为 具有 列 长 度 的 字符 串 ,车 不 够 指定 的 长 度 , 则 使 用 空格 填充 它 。 相 反 , VARCHAR 字 
符 串 只 与 存储 的 字符 串 本 身 一 样 长 。 

这 意味 着 : 

。 VARCHAR 列 倾向 于 占用 较 少 的 磁盘 空间 。 

。 除非 正在 使 用 InnoDB 表 类 型 ,否则 CHAR 列 的 访问 速度 比 VARCHAR 列 要 快 

一 些 。 

当然 ,在 大 多 数 情况 下 无 法 感知 这 两 种 类 型 之 间 的 速度 和 磁盘 空间 差别 ,并 且 随 着 
InnoDB 表 类 型 变 成 一 种 规范 ,任何 速度 差别 都 会 消失 。 

这 两 种 类 型 之 间 还 有 第 三 种 细微 的 差别 ,在 检索 数据 时 ,MySQL 会 删除 CHAR 列 
中 多 余 的 空格 ,而 对 于 VARCHAR 列 , 这 种 操作 则 发 生 在 插入 数据 时 。 

如 果 字 符 串 字段 总 是 具有 设置 的 长 度 ( 例 如 ,状态 简写 ) , 则 使 用 CHAR; 和 否则 ,使 用 
VARCHAR。 不 过 ,在 某 些 情况 下 ,MySQL 把 某 一 列 定义 为 一 种 类 型 (如 CHAR) ,即使 
把 它 创建 成 男 一 种 类 型 (VARCHAR) 也 是 如 此 。 这 很 正常 ,并 且 是 MySQL 提高 性 能 的 
一 种 方式 。 

多 种 不 同 的 日 期 类 型 具有 各 种 独特 的 行为 ,MySQL 手册 中 有 关于 它们 的 文档 。 我 
们 将 不 加 修改 地 使 用 DATE 和 TIME 字段 ,因此 ,不 必 过 于 关心 它们 的 错综复杂 的 情况 。 

还 有 两 种 特殊 类 型 一 -ENUM 和 SET 一 一 允许 为 字段 定义 一 系列 可 接受 的 值 。 
ENUM 列 只 能 从 数 千 个 可 能 的 值 中 取 一 个 值 ,而 SET 允许 从 最 多 64 个 可 能 的 值 中 取 多 
个 值 。 它 们 在 MySQL 中 都 是 可 用 的 ,但 是 并 非 每 一 种 数据 库 应 用 程序 都 支持 它们 。 

另外 ,还 有 一 个 BLOB 类 型 , 它 是 TEXT 的 一 个 变 体 ,允许 在 表 中 存储 二 进 制 文件 
(如 图 像 )。 这 种 类 型 还 用 于 某 些 加 密 数据 。 

2. 选择 列 类 型 

(1) 确定 某 一 列 应 该 是 文本 、 数 字 , 还 是 日 期 类 型 (参见 表 8-3) 。 

这 一 步 通常 比较 容易 和 明显 。 若 在 文本 字段 中 发 现 诸如 邮政 编码 和 货币 金额 之 类 的 
数字 ,其 中 包括 了 它们 相应 的 标点 符号 (美元 符号 .逗号 和 连 字 符 ), 但 是 把 它们 存储 为 数 
字 并 在 别 的 地 方 处 理 格式 化 效果 , 则 将 得 到 更 好 的 结果 。 

(2) 为 每 一 列 选择 最 合适 的 子 类 型 (参见 表 8-4) 。 

在 这 个 示例 中 ,将 把 user_id 设置 为 INT, 从 而 允许 多 达 接 近 430 万 个 值 ( 如 无 符号 
数 或 非 负 数 ) 。registration_date 将 是 DATETIME, 存 储 用 户 注 册 那 一 天 的 日 期 和 特定 
的 时 刻 。 当 在 多 种 日 期 类 型 之 间 做 出 抉择 时 ,要 考虑 是 只 想 访问 日 期 或 时 间 ,还 是 两 者 可 
能 都 想 访问 。 
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表 8-3 具有 泛 型 数据 类 型 的 users 表 表 8-4 具有 更 具体 数据 类 型 的 users 表 
列 名 类 型 列 名 类 型 
user_id 数字 user_id INT 
name 文本 name VARCHAR 
password 文本 password CHAR 
email 文本 email VARCHAR 
registration_date 日 期 /时 间 registration_date DATETIME 


其 他 字段 主要 是 VARCHAR, 因 为 它们 的 长 度 因 记录 而 异 。 唯 一 的 例外 是 密码 , 它 
是 定 长 的 CHAR, 这 是 因为 将 密码 通过 加 密 后 再 保存 至 数据 库 。 
(3) 设置 文本 和 数字 列 的 最 大 长 度 (参见 表 8-5)。 
表 8-5 设置 了 长 度 属性 的 users 表 


列 名 类 型 列 名 类 型 
user_id INT email VARCHAR(50) 
name VARCHAR(20) registration_date DATETIME 
password CHAR(40) 


应 该 基于 可 能 最 大 的 输入 ,把 任意 字段 的 大 小 都 限制 为 尽 可 能 小 的 值 。 例 如 ,如 果 最 
大 的 数 (比如 user_id) 可 能 是 几 百 , 则 把 该 列 设置 成 3 位 的 SMALLINT( 人 允许 最 多 999 个 


值 ) 。 
3. 完成 列 的 定义 


在 设计 数据 库 时 ,需要 考虑 创建 索引 、 添 加 键 以 及 使 用 AUTO_INCREMENT 属性 。 


(1) 确定 主键 。 


主键 可 随意 选择 但 特别 重要 。 一 般 来 说 , 它 就 是 一 个 数 , 可 用 来 引用 特定 记录 。 例 
如 ,电话 号 码 没有 固定 值 , 但 它 是 联系 的 独特 方式 。 
在 users 表 中 ,user_id 将 作为 主键 : 它 是 用 于 引用 一 行 数据 的 任意 数字 。 
(2) 确定 哪些 列 不 能 有 NULL 值 。 
在 这 个 示例 中 ,每 个 字段 都 是 必需 的 , 即 不 能 是 NULL。 与 之 相对 ,如 果 要 存储 地 


址 , 则 可 能 有 address_linel 和 address_line2 两 列 , 其 中 后 者 是 可 选 的 。 


(3) 如 果 一 个 数字 类 型 永远 不 会 存储 负数 , 则 可 将 其 设置 成 UNSIGNED 无 符号 数 。 
user_id( 表 示 一 个 数字 ) 是 UNSIGNED., 它 总 为 正 。 
(4) 为 任何 列 创 建 默认 值 。 
此 处 所 有 列 在 逻辑 上 都 没有 隐 含 一 个 默认 值 。 


(5) 确认 最 终 的 列 定义 (参见 表 8-6) 。 


在 创建 表 之 前 ,应 该 复查 将 要 存储 的 数据 的 类 型 和 范围 ,确保 让 数据 库 有 效 地 涵盖 各 


个 方面 的 情况 。 
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表 8-6 users 表 的 最 终 描述 ,其 中 user_id 将 被 标记 为 自动 递增 的 主键 


列 名 类 型 列 名 类 型 

user_id INT UNSIGNED NOT NULL |email VARCHAR(50) NOT NULL 
name VARCHAR(20) NOT NULL |registration_date | DATETIME NOT NULL 
password CHAR(40) NOT NULL 


在 users 表 中 ,user_id 将 被 指定 为 PRIMARY KEY( 主 键 ), 它 既是 列 的 描述 ,又 指示 
MySQL 对 其 加 索引 。 因 为 user_id 是 一 个 数字 (主键 几乎 总 是 数字 ) ,所 以 还 对 该 列 添加 
了 AUTO_INCREMENT 描述 , 它 告 诉 MySQL 使 用 下 一 个 最 大 的 数字 作为 每 一 条 添加 
记录 的 user_ id 值 。 


8.2 操作 MySQL 


MySQL 是 世界 上 最 流行 的 开源 数据 库 应 用 程序 (依据 MySQL 的 Web 站 点 www. 
MySQL. com) ,并 且 通 常 与 PHP 一 起 使 用 。MySQL 软件 带 有 数据 库 服 务 器 .不 同 的 客 
户 应 用 程序 以 及 多 种 实用 程序 。 


8.2.1 使 用 命令 行 管理 MySQL 


为 了 创建 表 、 添 加 记录 以 及 从 数据 库 请 求 信息 ,需要 某 种 客户 应 用 程序 与 数据 库 服务 
器 通信 。 尽 管 有 许多 客户 应 用 程序 可 用 ,这 里 先 介 绍 如 何 使 用 MySQL 客户 端 。 尽 管 这 
种 应 用 程序 没有 漂亮 的 图 形 界面 ,但 它 是 一 种 可 靠 的 标准 工具 ,易于 使 用 ,并 且 在 许多 不 
同 的 操作 系统 上 具有 一 致 的 工作 方式 。 

可 以 从 命令 行 界面 访问 MySQL 客户 端 ,在 类 UNIX 中 ,命令 行 界面 是 终端 应 用 程 
序 ; 在 Windows 中 , 则 是 DOS 提示 符 。 它 可 以 预先 带 有 几 个 参数 ,包括 用 户 名 、 密 码 和 主 
机 名 (计算 机 名 或 URL)。 可 以 按 如 下 方式 建立 这 些 参 数 : 


mysql] -u username -P -h hostname 


-p 选项 将 导致 客户 端 提示 输入 密码 。 同 时 ,也 可 以 在 这 一 行 指定 密码 ,直接 在 -p 提 
示 符 后 面 输入 它 , 但 是 它 将 是 可 见 的 ,这 不 安全 。-h hostname 参数 是 可 选 的 ,建议 不 要 使 
用 它 ,除非 不 能 以 别 的 方式 连接 到 MySQL 服务 器 。 

在 MySQL 客户 端 内 ,需要 用 分 号 来 终止 每 一 条 语句 ,也 就 是 SQL 命令 。 这 些 分 号 
指示 MySQL 查询 已 完成 。 分 号 不 是 SQL 自身 的 一 部 分 。 这 意味 着 在 MySQL 客户 端 
内 ,可 以 把 同一 条 SQL 语句 分 布 在 多 行 上 ,以 使 它 更 易于 阅读 。 

在 操作 MySQL 客户 端 执行 这 些 步骤 之 前 : 

。 必须 运行 MySQL 服务 器 。 

。 必须 有 用 户 名 和 密码 .它们 具有 适当 的 访问 权限 。 
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(1) 从 命令 行 界面 访问 MySQL 

在 类 UNIX 系统 上 ,要 从 命令 行 界面 访问 MySQL 客户 端 , 这 需要 启动 终端 。 如 果 正 
在 使 用 Windows 系统 , 则 可 以 选择 “开始 ”>“ 程 序 ”>MySQL~>MySQL Server X. X 一 
MySQL Command Line Client 命令 MySQL Windows 安装 程序 会 在 “开始 ”菜单 中 创 
建 一 个 链接 ,使 得 可 以 轻松 进入 MySQL 客户 端 

(2) 使 用 合适 的 命令 调用 MySQL 客户 端 


/path/to/mysql/bin/mysql -u username -p 


这 一 步 中 的 /path/to/mysql 部 分 主要 是 由 正在 运行 的 操作 系统 以 及 MySQL 的 安装 
位 置 指定 的 。 因 此 , 它 可 能 是 : 


/usr/local/mysql/bin/mysql -u username -p( 在 类 UNIX 上 ) 
或 
C:\mysql\bin\mysql -u username -p( 在 Windows 上 ) 


基本 的 前 提 条 件 是 ,正在 运行 MySQL 客户 端 ,以 username 连接 ,并 请 求 被 提示 输入 
密码 。 如 果 具 有 相应 访问 信息 ,可 以 使 用 root 用 户 , 但 是 有 权 创 建 表 和 数据 库 的 任何 用 
户 都 是 可 行 的 。 

(3) 登录 MySQL 客户 端 

在 提示 符 后 面 输入 密码 ,并 按 下 回 车 键 

在 这 里 使 用 的 密码 应 该 对 应 于 在 上 一 步 指定 的 用 户 。 如 果 使 用 Windows 上 的 
MySQL Command Line Client 链接 , 则 用 户 是 root, 因 此 ,应 该 使 用 它 的 密码 ,这 个 
可 能 是 在 安装 和 配置 期 间 建立 的 。 

如 果 使 用 正确 的 用 户 名 /密码 组 合 , 即 某 个 具有 有 效 访问 权限 的 人 , 则 应 该 看 到 如 
图 8-1 所 示 的 问候 。 


图 8-1 如 果 能 够 成 功 地 登录 , 则 会 看 到 如 图 中 所 示 的 欢迎 消息 
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(4) 选择 想 使 用 的 数据 库 , 如 图 8-2 所 示 


use test; 


图 8-2 USE 命令 用 于 选择 要 使 用 的 数据 库 


use 命令 告诉 MySQL 从 现在 起 用 户 想 处 理 哪 个 数据 库 , 从 而 避免 以 后 反复 输入 数据 
库 名 称 。test 数据 库 是 MySQL 会 默认 安装 的 数据 库 之 一 。 假 定 它 存在 于 服务 器 上 ,所 
有 用 户 都 应 该 能 够 访问 它 

(5) 退出 MySQL。 

输入 exit 或 quit, 终 止 会 话 并 离开 MySQL 客户 端 。 如 果 使 用 MySQL Command 
会 自动 关闭 DOS 提示 符 窗口 

大 多 数 系统 上 的 MySQL 允许 使 用 向 上 和 向 下 的 箭头 键 来 滚动 以 前 输入 的 命令 。 在 
处 理 数 据 库 时 ,这 可 以 节省 许多 时 间 。 

如 果 正 在 执行 一 条 较 长 的 语句 并 且 犯 了 一 个 错误 ,可 以 输入 c 并 按 下 回 车 键 来 取消 
当前 的 操作 。 如 果 MySQL 认为 遗漏 了 一 个 右 单 引号 或 双 引 号 (如 这 和 "二 提示 符 所 提示 
的 那样 ) , 则 需要 首先 输入 合适 的 引号 。 


Line Client ， 


8.2.2 用 phpMyAdmin 管理 MySQL 


phpMyAdmin 是 一 种 用 PHP 编写 的 流行 的 开源 工具 , 它 为 MySQL 提供 了 一 个 基 
于 Web 的 界面 。 可 以 从 www. phpmyadmin. net 得 到 phpMyAdmin ,该 软件 非常 普及 ， 
许多 Web 托管 公司 将 之 作为 其 用 户 连 接 MySQL 数据 库 的 默认 方式 。 

phpMyAdmin 是 众多 MySQL 图 形 化 管理 工具 中 应 用 最 广泛 的 一 种 , 它 是 基于 PHP 
语言 编写 的 ,该 工具 是 B/S 结构 的 ,基于 Web 跨 平台 的 管理 程序 ,并 且 支 持 简体 中 文 。 
安装 后 在 浏览 器 地 址 栏 输入 “http://localhost/phpMyAdmin/” 即 可 进入 MySQL 的 管理 
界面 。 
phpMyAdmin 为 Web 开发 人 员 提 供 了 类 似 于 Access、SQL Server 的 图 形 化 数据 库 
操作 界面 ,通过 该 管理 工具 可 以 进行 绝 大 部 分 的 MySQL 操作 ,包括 对 数据 库 及 数据 表 的 
建立 和 维护 。 
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无 论 是 Windows 操作 系统 还 是 Linux 操作 系统 下 ,phpMyAdmin 图 形 化 管理 工具 
的 使 用 方法 都 是 相同 的 。 


1. 登录 


打开 Web 浏览 器 ,在 地 址 栏 输入 http://localhost/phpMyAdmin/, 此 时 会 弹出 登录 
对 话 框 ,提示 输入 登录 MySQL 的 用 户 名 和 密码 ,如 图 8-3 所 示 。 


图 8-3 ”用户 登录 对 话 框 


在 登录 phpMyAdmin 工具 时 ,输入 正确 的 用 户 名 和 密码 。 在 安装 MySQL 时 ,设置 
了 MySQL 数据 库 的 用 户 名 为 root 及 其 密码 。 输 入 后 即 可 进入 phpMyAdmin 的 主 界 面 ， 
如 图 8-4 所 示 。 


https//localhost/phpmyadmin/ 起 localhost / localhost | ph-. * 
phpMyAdmin - 2.10.3 


» MySQL client version: 5.0.51a 
» Used PHP extensions: mysql 


， 用 户 : root@glocalhost 
加 MySQL 字符 集 .UTF-8 Unicode (utf8) 
加 MySQL 连接 权 对 : 
Uf8_unicode ci -@ 
名 创建 一 个 新 的 数据 库 回 


Er 
章 显 示 MySQL 的 运行 信息 
国 显示 MySQL 的 系统 变量 @ 


图 8-4 phpMyAdmin 的 管理 界面 
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phpMyAdmin 的 管理 界面 分 为 三 大 块 . 左 侧 为 数据 库 选 择 区 域 , 中 间 为 数据 库 创 建 
区 域 , 右 侧 为 bbpMyAdmin 的 配置 区 域 。 我 们 的 主要 工作 都 集中 在 左 .中 两 块 区 域 。 


2. 库 级 操作 

一 切 准备 就 绪 , 接 下 来 就 可 以 进行 MySQL 管理 操作 了 ,首先 介绍 如 何 建立 和 删除 数 
据 库 。 

1) 创建 数据 库 

在 phpMyAdmin 的 主 界面 中 有 两 个 文本 框 和 “创建 ”按钮 ,首先 在 文本 框 中 输入 数据 
库 的 名 称 ,然后 选择 编码 ,最 后 单 击 “创建 "按钮 ,这 样 新 的 。 证- 有 话 包 
数据 库 就 可 以 被 创建 成 。 例 如 : 创建 一 个 名 称 为 db_test laotest | 
的 数据 库 , 首 先 在 文本 框 中 输入 db_test, 之 后 在 下 拉 列 表 的 7 
框 中 找到 要 使 用 的 字符 集 编码 ,在 Windows 下 一 般 选择 
gb2312_Chinese_ci, 如 图 8-5 所 示 。 

最 后 单 击 “ 创 建 "按钮 ,这 样 名 为 db_test 的 数据 库 就 被 创建 成 功 , 结 果 如 图 8-6 所 示 。 


8-5 创建 一 个 新 的 数据 库 


图 服务器: localhost ， 辑 数据 库 : db_test 
_ 队 结构 ”加 SQL 万 搜索 忆 查 询 范 导 出 而 Imporl 屁 换 作 蚁 权限 四 开除 


人 ee 数据 库 db lest 已 经 建立。 
DMS gw 
数据 库 CREATE DATABASE “db_test DEFAULT CHARACTER SET gb2312 COLLATE gb2312_chinese_d 
db test (0) 司 
一 一 一 一 | [ 纺 回 J[ 创 建 PHP 代码 ] 
db_test (0) 
[i 插话 中 没有 家 。 


三 罚 在 数据 库 db_test 中 创建 一 个 新 表 - 
全 字 : Number of fields: 


| 


图 8-6 数据 库 的 建立 


该 数据 库 名 称 出 现在 左 侧 导 航 栏 的 数据 库 下 拉 菜 单 中 , 单 击 选择 这 个 数据 库 ,在 右 侧 
界面 中 可 以 对 该 数据 库 进 行 操作 ,如 结构 .SQL、 导 出、 搜索 查询、 删除 , 单 击 相应 的 按钮 
即 可 进入 相应 的 操作 界面 。 

在 右 侧 界面 还 可 以 执行 创建 数据 表 的 操作 ,只 要 在 创建 数据 表 的 提示 信息 下 面 的 两 
个 文本 框 中 分 别 输入 要 创建 的 数据 表 的 名 称 和 字段 总 数 . 然 后 单 击 “ 执 行 ”按钮 即 可 进入 
到 创建 数据 表 的 页 面 。 

2) 删除 数据 库 

要 删除 某 个 数据 库 , 首 先 在 左 侧 的 下 拉 菜 单 中 选择 该 数据 库 , 然 后 单 击 右 侧 界面 中 的 
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“删除 ”按钮 即 可 。 系 统 会 弹出 对 话 框 要 求 确认 删除 操 


来 自 网 页 的 消息 
作 , 如 图 8-7 所 示 。 
3. 表 级 操作 @ HE es, 
针对 表 级 操作 是 在 选 定 了 数据 库 的 情况 下 进行 ee 
的 , 即 表 级 操作 的 前 提 是 必须 选择 了 一 个 数据 库 , 在 该 
数据 库 中 进行 表 的 建立 和 维护 。 [me] [iu 
1) 创建 表 


-7 数据 库 的 建立 
创建 数据 库 db_test 后 .在 右 侧 页 面 中 会 出 现 如 


图 8-6 所 示 的 数据 表 创 建 提示 页 面 。 首 先 在 表单 中 输入 数据 表 的 名 称 和 字段 数 , 如 图 8-8 


所 示 。 
性 在 数据 库 db_test 中 创建 一 个 新 表 
名 字 :|user Number of fields: 3 


8-8 创建 表 信息 


然后 单 击 “执行 "按钮 ,进入 各 个 字段 的 详细 信息 录入 表单 ,包括 字段 名 数据 类 型 .长 
度 / 值 .属性 ,默认 值 额外 和 索引 的 类 型 等 。 这 样 就 完成 了 对 表 结 构 的 详细 设置 ,如 图 8-9 
所 示 。 


图 服务 器 : localhost ， 数据 库 : db_test ， 国 表 : user 

子 腿 类 型 加 长 度 基 -4 整理 屋 性 
Im 司 m [| 司 C== 9 
Phame | [Wacmra 辐 [| [e232omneseo 辐 呈 ” 司 
asswou [cnwR 司 区 J[ 司 [ 避 


表 广 释 : Storage Engine: @ 整理 : 
网 户 二 名表 [mssmm ” 司 [ 司 


保存 | 或 添加 | 字段 执行 | 
图 8-9 输入 字段 信息 


当 所 有 的 信息 都 输入 以 后 ,就 可 以 单 击 “保存 ?按钮 ,这 时 将 执行 CREATE TABLE 
语句 ,成功 则 会 生成 如 图 8-10 所 示 的 界面 。 

一 个 新 的 数据 表 被 创建 后 .进入 到 数据 表 页 面 中 .在 这 里 可 以 通过 改变 表 的 结构 来 修 
改 表 ,可 以 执行 添加 新 的 列 、 删 除 列 、 索 引 列 ,修改 列 的 数据 类 型 或 者 字段 的 长 度 / 值 等 
操作 。 

2) 删除 表 

执行 删除 表 的 操作 很 简单 ,只 要 单 击 页 面 上 方 的 “删除 ”按钮 ,就 可 以 轻松 地 删除 一 个 
数据 表 , 即 执行 一 次 DROP TABLE 命令 .如 图 8-11 所 示 。 
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表 user 已 经 建立 。 


厂 “password varchar(20) gb2312_chinese_ci 否 


企 _ 全 选 /全 部 不 选 过 办 允 国 少 X 国 加 加 
疗 打印 预览 _ 唱 关系 查看 蝇 规划 表 结构 图 
怠 添 加 | 字段 C 于 未 结尾 C 于 表 开 头 C 于 lid 司 之 后 执行 | 


素 引 : 回 已 使 用 空间 行 统计 
键 名 大 型 ”基数 操作 字段 ”类 型 用 法 语句 值 


PRIMARY PRIMARY 0 FX 数据 0 字 节 
第 6 峙 素 引 下 | rr 


8-10 成 功 创建 表 


您 真 的 要 : 
DROP TABLE ‘users™ 


Ce Cw 


8-11 提示 删除 


4. 增删 改 数据 

数据 库 和 数据 表 的 创建 只 能 算是 达到 某 个 目的 的 手段 ,而 数据 才 是 使 用 这 些 手段 要 
实现 的 目的 。 没 有 了 数据 的 数据 表 和 数据 库 可 以 说 是 没有 任何 意义 的 。 如 果 缺 少 了 这 些 
手段 ,数据 也 就 不 存在 了 。 

所 以 .在 对 数据 进行 操作 之 前 ,首先 要 选择 一 个 数据 表 , 即 单 击 左 侧 的 数据 表 , 在 右 侧 
的 页 面 中 就 会 出 现 该 数据 表 的 信息 。 插 入 数据 有 两 种 方法 .下面 分 别 介绍 。 

1) 通过 SQL 语句 来 操作 数据 

单 击 界面 中 的 SQL 按钮 ,就 可 以 打开 SQL 语句 编辑 区 。 在 编辑 区 输入 完整 的 SQL 
语句 ,来 实现 数据 的 查询 添加、 修改 和 删除 操作 。 为 了 编写 方便 ,还 可 以 利用 其 右 侧 的 属 
性 列表 来 选择 要 操作 的 列 ,只 要 选中 要 添加 的 列 , 然 后 单 击 " 二 二 ”按钮 即 可 ,如 图 8-12 
所 示 。 

在 语句 编写 完成 后 , 单 击 “ 执 行 ”按钮 提交 .如 果 提 交 的 SQL 语句 有 错误 ,系统 会 给 出 
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一 个 警告 ,提示 用 户 修改 它 , 如 图 8-13 所 示 。 


广 在 数据 库 db _test 运行 SQL 查询 : @ 


司 字段 数 

四 

name 
password 


4 到 
将 此 SQL 查询 加 为 书签 - 厂 让 每 个 用 户 都 可 以 访问 这 个 书签 


厂 Replace existing bookmark of same 
name 


[Delimiter[ ””“] F 在 此 再次 显示 此 查询 


图 8-12 ”SQL 语句 的 输入 框 


胃 服务 器 : localhost ， 加 数据 库 : db_test ， 回 表 : user " 历 户 入 息 夫 " 
错误 


SQL 查询 : 国 


SELECT* 

FROM “user 
WHERE 'emair = Wan@126.com' 
UMT0 .30 


MySQL 返回 : 国 


#1054 - Unknown column "email' in "where clause" 


图 8-13 SQL 错误 


2) 通过 单 击 “ 插 入 "按钮 添加 数据 

可 以 单 击 页 面 上 方 的 “插入 ”按钮 ,进入 数据 添加 页 面 ,在 该 页 面 中 只 需 在 对 应 的 文本 
框 中 输入 数据 ,然后 单 击 “ 执 行 " 按 钮 即 可 ,如 图 8-14 所 示 。 

5. 查询 数据 

1) SQL 查询 

对 于 数据 的 查询 ,可 以 使 用 上 面 介绍 的 SQL 语句 实现 ,编写 完整 的 SQL 语句 ,例如 
查询 整个 表 的 信息 ,代码 如 下 。 


select * from user; 


这 是 对 整个 表 的 简单 查询 ,还 可 以 进行 一 些 复 杂 的 条 件 查 询 , 使 用 WHERE 子 句 提 
交 LIKE、ORDER BY、GROUP BY 等 条 件 查询 语句 。 例 如 ,查询 user 表 中 用 户 名 中 有 
“Mary” 字 符 串 的 用 户 。 代 码 如 下 。 


SELECT #* FROM 'user' WHERE 'name'= "Mary" 
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困 服务 器 : localhost ， 昌 数据 库 : db_test ， 国 表 : user " 历 户 信息 起 " 
_ 国 浏览 办 结 构 妨 SQl 万 搜索 著 疾 入 障 导 出 并 Impor 做 要 作 [而 法 宇 网 到 除 


[095A 司 sxen [局 好 上 不 本 
9 [ES 


图 8-14 输入 数据 表 记 录 
2) 通过 “浏览 "和 “搜索 ”按钮 查询 
可 以 单 击 * 浏 览 ?按钮 进入 数据 表 详 细 信 息 页 面 , 如 图 8-15 所 示 , 能 够 看 到 表 的 详细 
信息 。 


显示 行 0- 1 (2 总 计 , 查询 花费 0.0003 秒 ) 


[编辑 ][ 解 释 SQL] [创建 PHP 代码 ][ 刷 新 ] 
Query results operations 
学 打印 预览 ”部 打印 预览 (全 文 显示 ) ” 瞄 导出 


EE ,mg: 
以 [水 后 ” ”” 辐 模 式 显示 ,并 且 在 [100 个 单元 格 后 重复 标 是 
主键 排序 -无 司 邯 到 
全 一 idaess namexs password == 
x 1 May 123456 


| 
企 _ 全 选 /全 部 不 选 首 史 项 几 X 圈 
Bo ,mR: D 


以 | 水平 司 禄 式 旺 示 ， 并 且 在 [100 。 个 单元 格 后 重复 标题 


图 8-15 数据 表 信 息 


通过 单 击 页 面 中 的 “搜索 ”按钮 ,可 以 进入 到 条 件 查 询 页 面 ,在 这 个 页 面 中 ,可 以 在 选 
择 字段 的 列表 框 中 选择 一 个 或 多 个 列 , 如 果 要 选择 多 个 列 , 先 按 下 Ctrl 键 并 单 击 要 选择 
的 列 即 可 ,如 图 8-16 所 示 。 

这 里 给 出 两 种 查询 方式 。 第 一 种 方式 可 以 选择 构建 WHERE 子 句 ,直接 在 “where 
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园 服务 器 : localhost ” 数据 库 : db_test ， 国 表 : user " 历 户 售 廊 过 " 
_ 转 训 览 “了 困 结 构 ， SQL “万 搜索 也: 括 入 肛 导 出 了 关 Imporl 贷 换 作 国清 空 加 到 除 


| 广 每 页 行 数 
至 少 选择 一 个 字段 : Bo- 
-. 
厂 显示 顺序 
[ae 悦 
人 递增 6 甫 请 


添加 搜索 条 件 (“where” 语 句 的 主体 )， 回 


厂 有 执行 “ 按 例 查询 ”( 通 配 符 ; “%”) 
字 有 & 类 型 得 
a tt10) EE 司 k | 
| 
password 5 9%2312_ chmesed [rE 司 站 
) 


| 


8-16 ”搜索 查询 


语句 的 主体 ”文本 框 中 输入 查询 语句 ,然后 单 击 其 后 的 “执行 "按钮 。 第 二 种 方式 可 以 使 用 
“ 按 例 查询 ”下 的 文本 框 ,在 文本 框 中 选择 要 查询 的 条 件 和 输入 要 查询 的 值 ,或 选择 查询 结 
果 的 排序 方式 。 在 所 有 条 件 都 添加 完成 后 , 单 击 “ 执 行 "按钮 。 

6. 数据 的 导入 与 导出 

使 用 图 形 化 管理 工具 可 以 进行 数据 的 导入 和 导出 ,同时 也 可 以 对 整个 的 数据 表 或 数 
据 库 执行 此 操作 。 

1) 数据 的 导入 

单 击 import 按钮 ,就 可 以 进行 文件 导入 了 , 单 击 “ 浏 览 ” 按 钮 查找 SQL 文件 所 在 位 
置 ,然后 单 击 “ 执 行 " 按 钮 即 可 ,如 图 8-17 所 示 。 

在 导入 文件 过 程 中 ,如 果 导 入 的 是 一 个 完整 的 数据 库 文件 ,首先 要 在 数据 库 中 创建 一 
个 名 称 与 数据 文件 中 的 数据 库 名 相同 的 数据 库 , 然 后 再 进行 导入 文件 的 操作 。 

如 果 导 入 的 是 某 个 数据 库 中 的 一 个 完整 的 数据 表 , 只 需 在 左 侧 的 数据 库 列表 中 选择 
该 数据 库 , 就 可 以 进行 导入 文件 的 操作 了 。 在 该 数据 库 中 不 能 有 与 将 要 导入 数据 库 中 的 
数据 表 重 名 的 数据 表 存 在 ,如 果 有 重 名 的 表 存 在 导入 文件 就 会 失败 ,提示 错误 信息 一 一 该 
数据 表 已 经 存在 。 

如 果 导 入 的 数据 文件 只 是 一 些 数据 ,不 包含 建立 数据 表 的 部 分 ,就 需要 在 数据 库 中 建 
立 一 个 对 应 的 数据 表 , 包 括 字 段 、 属 性 、 类 型 和 默认 值 这 些 都 必须 是 对 应 的 ,否则 导入 文件 
的 操作 不 会 成 功 。 
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图 服务 器 : localhost ， 电 数据 库 : db_test 
_ 因 站 构 ”加 SQL 万 搜索 局 查询 项 导 出 随 Impot 和 操作 鸣 权 限 网 到 除 


Import 
File to import 
文本 文件 的 位 置 |C:\ch08Wb_test sql 浏览 | (最 大 限制 : 2,048 KB) 
文件 的 字符 集 : |utf8 | 


Imported file compression will be automatically detected from: 无, gzip, zip 


Partial import 
Allow interrupt of import in case script detects it is close to time limit This might be good 
‘Way to import large files, however it can break transactions- 


Number of records(queries) to skip from start |0 


8-17 从 SQL 文件 中 提取 数据 导入 


2) 数据 的 导出 

可 以 通过 单 击 右 侧 页 面 中 的 “导出 ”按钮 ,进入 到 如 图 8-18 所 示 的 页 面 来 完成 操作 。 
导出 文件 支持 多 种 格式 ,包括 SQL、LaTeX、XML 等 。 可 以 选择 直接 在 该 页 输出 数据 , 那 
就 不 用 选中 “另存 为 文件 " 复 选 框 ,直接 单 击 “执行 "按钮 即 可 ;如 果 要 将 文件 另存 到 某 个 文 
件 夹 中 ,就 要 选中 “另存 为 文件 " 复 选 框 ,然后 单 击 “ 执 行 ” 按 钮 ,选择 要 保存 在 什么 位 置 中 
并 给 文件 命名 。 

还 可 以 使 用 “压缩 * 复 选 框 中 的 压缩 类 型 来 压缩 导出 的 文件 ,便于 处 理 较 大 的 表 。 

具体 导出 文件 的 格式 如 下 : 


——PhpMyAdmin SQL Dump 

-—version 2.9.2 
——http://www.phpmyadmin.net 

=- 主机: localhost 

-- 生 成 日 期 : 2014 年 08 月 07 日 07:09 
- -服务 器 版 本 : 5.0.27 

--BPHP 版 本 : 5.2.1 


-- 数 据 库 : 'db_test' 
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广 查 看 数据 库 的 转 存 (大 网 )。 
一 导出 厂 SQL 选项 
在 标题 增加 一 个 定制 的 注释 (wn 行 限 训 符 ) 
厂 处 理事 务 中 寺 装 输出 
厂 禁 上 过 定 不 相关 的 主键 
SQL compatibility mode 
NoNE 司 
人 造 /全 部 不 过 图 
G3 
三民 结构 
C MSExcel 的 厂 Add DROP TABLE /DROP VIEW 
CSV 格 式 厂 AddiFNOTEMSTS 
Microsoft 区 添加 AUTO_INCREMENT 值 
Excel 2000 请 在 表 名 及 字段 名 使 用 引号 
i 天 加 进 注释 
i 厂 外 建 厦 新 卓 期 检查 
厂 关系 
C LaTex 厂 MME 关 型 
CDpen 
Document 
Spreadsheet dd 
厂 完 吾 拍 入 
Open 厂 扩展 白 入 
Document Text Maximal length of created query 
A |50000 
厂 王 时 白 入 
G soL 厂 知 喇 拔 入 
二 进 币 区 域 使 用 十 六 进 制 显示 
C XML 生出 并 到 
INsERT 辐 
厂区 宪 为 文 信 
文件 名 模板 (让 五 DB ( F 记 住 模板 ) 
压缩 :6 无 人 zp 压缩 “gzip 压缩 


图 8-18 导出 界面 
-- 表 的 结构 "user' 


CREATE TABLE "user'( 
"id' int (10)unsigned NOT NULL auto_increment COMMENT ' 用 户 编号 '， 
'name' varchar (20)NOT NULL COMMENT ' 姓 名 '， 
"password' varchar (20)NOT NULL COMMENT ' 密 码 '， 
PRIMARY KEY ("id') 
)ENGINE=MyISAM DEFAULT CHARSET= gb2312 COMMENT= ' 用 户 信息 表 ' AUTO_INCREMENT=3 ; 


-- 导 出 表 中 的 数据 "user'" 


INSERT INTO "user' VALUES (1，'Mary'，"123456")7 

INSERT INTO "user' VALUES (2, 'Jack', '654321'); 

在 该 文件 中 ,以 “一 ”开头 的 行为 注释 的 内 容 。 在 脚本 中 包含 以 下 内 容 : 
phpMyAdmin 的 信息 、 服 务 器 和 导出 时 间 的 信息 数据库 的 信息 、 创 建 表 的 信息 及 添加 到 
表 中 的 数据 的 信息 。 
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8.3 SQL 基础 


SQL 语言 是 MySQL 服务 器 能 “ 听 懂 ”的 语言 ,要 与 MySQL 交互 ,就 必须 首先 了 解 
SQL。 在 使 用 诸如 MySQL 客户 端 这 样 的 程序 时 ,就 要 发 送 SQL 语句 给 服务 器 。 另 外 ， 
在 实现 数据 库 系 统 时 ,也 必须 熟悉 SQL 语言 ,通过 数据 库 接口 函数 ,如 MySQL_query() 
等 ,发 送 SQL 语句 与 数据 库 交互 。 

SQL(Structured Query Language, 结 构 化 查询 语言 ) 是 关系 数据 库 通 用 语言 ,其 地 位 
犹如 英语 在 世界 语言 中 的 地 位 。SQL 已 经 被 ANSI( 美 国 国家 标准 化 组 织 ) 确 定 为 数据 库 
系统 的 工业 标准 。 

SQL 的 通用 性 体现 在 : 在 不 同 的 数据 库 管 理 系统 上 .用户 可 以 用 几乎 同样 的 SQL 语 
名 执行 同样 的 操作 。 例 如 ,select * from [数据 表 名 ]” 从 某 个 数据 表 中 取出 全 部 数据 ， 
在 Oracle、SQL Server、Foxpro、MySQL .DB2 等 关系 型 数据 库 中 都 可 以 使 用 。 

1. SQL 标准 

在 E.F. Codd 博士 于 20 世纪 70 年 代 早 期 提出 关系 数据 库 理 论 之 后 不 久 就 创建 了 
SQL。1989 年 ,美国 国家 标准 协会 (ANSI) 一 一 负责 维护 该 语言 的 组 织 一 一 发 布 了 第 一 
个 SQL 标准 ,现在 称 之 为 SQL 89。SQL 2 于 1992 年 发 布 ,并 且 仍 然 是 当前 的 工作 版 本 ， 
也 称 为 SQL 92 或 者 简单 地 称 为 SQL。1999 年 ,ANSI 发 布 了 SQL 99 标准 ,也 称 SQL 3， 
但 迄今 主流 数据 库 均 未 完全 实现 该 标准 。 

2. SQL 语句 

按照 功能 ,SQL 语言 可 以 分 为 4 大 类 。 

(1) 数据 查询 语言 : 查询 数据 。 

(2) 数据 定义 语言 : 建立 删除 和 修改 数据 对 象 。 

(3) 数据 操纵 语言 : 完成 数据 操作 的 命令 ,包括 查询 。 

(4) 数据 控制 语言 : 控制 对 数据 库 的 访问 ,服务 器 的 关闭 .启动 等 。 

常用 的 SQL 命令 关键 字 如 表 8-7 所 示 。 

表 8-7 常用 的 SQL 命令 关键 字 


功能 分 类 SQL 关键 字 功 能 
CREATE/ALTER/DROP TABLE 创建 /修改 /删除 表 

数据 定义 语言 CREATE/ALTER/DROP VIEW 创建 /修改 /删除 视图 
CREATE/ALTER/DROP INDEX 创建 /修改 /删除 索引 
INSERT 向 表 中 插入 新 数据 行 

数据 操纵 语言 UPDATE 更 新 表 中 现 有 的 数据 
DELETE 从 表 中 删除 几 行 数据 

数据 查询 语言 SELECT 从 一 张 或 多 张 表 中 返回 数据 

EN GRANT 为 用 户 赋 予 特权 
村 阁 全 名 有 REVOKE 收回 用 户 特权 
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SQL 语言 简单 易学 、 风 格 统一 .利用 简单 几 个 英文 单词 的 组 合 就 可 以 完成 所 有 的 功 
能 。 在 MySQL 中 ,可 以 使 用 客户 端 直接 使 用 SQL 语言 ,这 些 语句 几乎 可 以 不 加 修改 地 
嵌入 到 PHP 中 。 利 用 PHP 的 计算 能 力 和 MySQL 的 数据 库 操纵 能 力 , 可 以 快速 建立 数 
据 库 应 用 程序 。 

作为 通用 的 关系 数据 库 语言 ,SQL 是 各 个 关系 数据 库 的 基础 ,同时 ,每 个 数据 库 往往 
都 会 对 其 有 一 些 扩展 ,以 支持 更 为 强大 的 功能 。 所 有 的 主流 数据 库 都 会 使 用 SQL， 
MySQL 也 不 例外 。SQL 有 多 种 版 本 ,而 且 MySQL 对 SQL 标准 的 实现 又 有 不 少 变异 ， 
但 是 SQL 仍然 极其 容易 学 习 和 使 用 。MySQL 支持 的 SQL 语句 如 表 8-8 所 示 。 


表 8-8 MySQL 支持 的 SQL 语句 


功 能 SQL 语句 
创建 .丢弃 和 选择 数据 库 CREATE/DROP DATABASE.USE 
创建 .更改 和 丢弃 表 和 索引 CREATE/ ALTER/DROP TABLE CREATE/ALTER/DROP INDEX 
从 表 中 选择 信息 SELECT 


取 数 据 库 、 表 和 查询 的 有 关 信 息 | DESCRIBE、EXPLAIN、SHOW 


DELETE、INSERT、UPDATE、LOAD DATA、 OPTIMIZE 


修改 表 中 信息 TABLE.REPLACE 
管理 语句 FLUSH.GRANT.KILL.REVOKE 


其 他 语句 CREATE/DROP FUNCTION .LOCK/UNLOCK TABLES SET 


同时 ,MySQL 还 对 标准 的 SQL 语句 进行 了 简化 , 即 它 并 非 支 持 所 有 的 SQL 语句 所 


8.4 MySQL 用 户 管理 


8.4.1 MySQL 管理 员 root 


数据 库 也 可 以 具有 多 个 用 户 ,就 像 操 作 系 统一 样 。MySQL 用 户 不 同 于 操作 系统 用 
户 , 即 使 他 们 共享 公共 名 字 。 因 此 ,MySQL 根 用 户 是 一 个 不 同 于 操作 系统 根 用 户 的 实 
体 , 他 们 具有 不 同 的 权力 ,甚至 具有 不 同 的 密码 (这 样 做 更 可 取 , 但 并 不 必要 ) 。 

MySQL 的 root 用 户 作 为 可 做 任何 事情 的 一 个 超级 用 户 被 创造 。 他 是 超 用 户 账户 ， 
可 以 执行 任何 操作 。 初 始 的 root 口令 是 空 的 ,因此 任何 人 都 能 以 root 而 没有 一 个 口令 进 
行 连接 并 且 被 授予 所 有 权限 。 在 安装 MySQL 的 过 程 中 ,安装 程序 会 要 求 配置 root 用 户 
的 密码 。 

MySQL 系统 内 的 每 位 用 户 都 具有 从 特定 主机 (计算 机 ) 访 问 特定 数据 库 的 特定 能 
力 。MySQL 根 用 户 具 有 最 大 的 权力 .并 且 用 于 创建 子 用 户 ,尽管 可 以 赋予 子 用 户 类 似 于 
根 用 户 的 权力 ,但 这 样 做 是 不 明智 的 。 

作为 MySQL 管理 员 , 应 该 知道 如 何 设置 MySQL 用 户 账号 ,指出 哪个 用 户 可 以 连接 


第 8 章 SQL 与 MySQL 


服务 器 ,从 哪里 连接 ,连接 后 能 做 什么 。 
8.4.2 用 户 管理 


当 一 位 用 户 试图 对 MySQL 服务 器 做 某 件 事情 时 ,MySQL 首先 会 检查 该 用 户 是 否 确 
实 有 权 连 接 到 服务 器 (基于 用 户 名 ,用 户 的 密码 和 MySQL 数据 库 的 用 户 表 中 的 信息 )。 
其 次 ,MySQL 将 会 检查 该 用 户 是 否 有 权 在 特定 的 数据 库 上 运行 特定 的 SQL 语句 ,例如 ， 
选择 数据 \ 插 入 数据 或 者 创建 新 表 。 为 了 确定 这 一 点 ,MySQL 使 用 db、host、user、tables 
_priv 和 columns 这 些 表 , 它 们 也 都 来 自 于 MySQL 数据 库 。 

在 MySQL 中 的 MySQL 系统 数据 库 的 user 数据 表 中 存 有 用 户 的 账号 信息 ,在 初始 
状态 下 已 存在 root 用 户 , 可 能 还 有 一 些 匿 名 用 户 , 且 所 有 用 户 都 没有 设置 密码 。 该 数据 
表 的 这 些 用 户 信息 是 通过 一 个 mysql_install_db 脚本 安装 的 。 

user 表 保存 所 有 可 以 访问 MySQL 的 用 户 , 如 图 8-19 所 示 , 它 的 主要 列 有 : 

。 Host, 允许 连 接 到 数据 库 服 务 器 的 主机 名 ,“%” 通 配 符 代表 所 有 主机 。 

。 User, 连 接 数 据 库 的 用 户 名 。 

。 Password, 连 接 密码 ,已 加 密 。 

。 其 他 权限 列 , 以 “Y” 或 “N” 标 识 是 否 有 效 。 

轩 服务 器 : localhost ”加 数据 库 : mysql ， 四 表 : user "Users and global privileges” 
浏览 。 [从 结构 “ 弄 SQL 万 搜索 革新 入 别 导 出 障 Import 性 据 作 名 清空 网 昌 除 


显示 行 0- 0 (1 总计, 查询 花费 0.0004 秒 ) 


SELECT 
FROM “User 
UMT0.30 
[编辑 ][ 逢 释 SQL] [创建 PHP 代码 ][ 出 新 ] 


Query results operations- 
[六 3 交 “ 委 打印 ( 文 & 示 ) 图 导 出 


Eo ,Fm p 
以 [水手 ” ”” 国 模 式 旺 示 , 并且 在 [100 个 单元 格 后 重复 标题 
全 一 Host User Password Select_priv Insert_priv Update_priv Delete_priv Create_priv Drop_pn 
厂 PX localhost root 565491d704013245 Y hy 5 种 A nt 


企 _ 全 选 /全 部 不 选 普 四 p 砚 及 X 罚 
五 寺 加 Po 。 行 ,开始 行 改 : | 
以 成 宙 国策 性 未 ,并 且 在 100 个 单元 格 后 重复 标题 


图 8-19 user 权限 表 


1. 添加 新 用 户 

当 管 理 员 root 添加 一 个 新 用 户 时 ,会 在 user 表 中 为 该 用 户 创建 一 条 记录 。 如 果 语 句 
指定 任何 全 局 权限 (管理 权限 或 适用 于 所 有 数据 库 的 权限 ), 那 么 这 些 也 记录 在 user 
表 中 。 

在 phpMyAdmin 管理 界面 单 击 “ 权 限 ” 按 钮 ,可 以 进入 “用 户 一 览 ” 界 面 ,其 中 列 出 了 
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所 有 能 登入 的 用 户 及 他 们 拥有 的 权限 ,如 图 8-20 所 示 。 


四 服务 器 : localhost 
_ 疝 数 大 库 . 玉 SQl 春 状 态 ” 同 变 景 “[ 阐 字符 集 、 因 Engines 区 权 限 ” 怕 进程 ” 羡 导 出 配 Import 


锡 用 户 一 览 
a 
用 户主 机 密码。 全 局 权限 四 授权 


厂 root localhost 是 ALL PRIVILEGES 是 Sp 
人 全 选 /全 部 不 选 


鲜 ”添加 新 用 户 


三 私 _ 出 除 选 中 用 户 
(收回 用 户 的 所 有 激活 权限 ， 然 后 删除 用 户 。 ) 
厂 删除 与 用 户 名 称 相同 的 数据 库 。 


| 


图 8-20 用 户 权限 浏览 


打开 链接 , 单 击 “ 添 加 新 用 户 " 链 接 , 在 打开 的 页 面 中 会 看 到 “登入 信息 ”, 输 入 用 户 名 、 
主机 (若是 本 地 , 则 为 localhost) ,密码 后 , 单 击 最 下 面 的 “执行 ”按钮 ,这 样 就 新 建 了 一 个 用 
户 名 。 

用 户 名 的 最 短 长 度 为 16 个 字符 。 在 创建 用 户 名 时 ,应 避免 使 用 空格 ,并 注意 用 户 名 
是 区 分 大 小 写 的。 密码 没有 长 度 限制 ,但 是 也 区 分 大 小 写 。 密 码 将 在 MySQL 数据 库 内 
进行 加 密 , 这 意味 着 不 能 以 明文 格式 恢复 它们 。 

其 中 ,还 一 个 选项 可 用 于 把 用 户 限定 到 特定 的 主机 名 。 主 机 名 可 以 是 运行 MySQL 
服务 器 的 计算 机 的 名 称 ( 在 这 里 ,localhost 是 最 常见 的 值 ) ,或 者 是 用 户 用 来 访问 服务 器 
的 计算 机 的 名 称 , 甚 至 可 以 是 一 个 IP 地 址 。 

在 一 般 情况 下 ,建议 给 单个 用 户 配 置 单个 数据 库 的 权限 ,所 以 在 这 里 不 选择 “全 局 权 
限 ” 选 项 组 中 的 各 个 选项 ,如 图 8-21 所 示 。 

这 里 我 们 成 功 创建 了 一 个 新 的 MySQL 连接 用 户 ,如 图 8-22 所 示 , 用 户 名 为 
testuser, 只 人 允许 从 本 地 localhost 连接 , 设 有 密码 。 未 授予 权限 ,将 在 以 后 将 具体 权限 授 
予 该 账户 。 

2. 删除 用 户 

要 使 用 删除 用 户 ,必须 拥有 MySQL 数据 库 的 全 局 CREATE USER 权限 或 
DELETE 权限 。 删 除 用户 ,意味 着 也 将 取消 这 个 账户 及 其 权限 。 

在 phpMyAdmin 中 ,删除 用 户 的 操作 将 不 会 弹出 任何 确认 提示 ,同时 会 提供 是 否 删 
除 与 用 户 名 称 相同 的 数据 库 的 选项 ,如 图 8-23 所 示 。 

删除 用 户 不 能 自动 关闭 任何 打开 的 用 户 对 话 。 而 且 ,如果 此 时 用 户 正在 连接 的 对 话 ， 
此 时 取消 用 户 , 则 命令 不 会 生效 ,直到 用 户 对 话 被 关闭 后 才 生 效 。 一旦 对话 被 关闭 ,用 户 
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Database for user- 


回 None 


© Create database with same name and grant all prvileges 
© Grant all privieges on wildcard name (usemame\ %) 


广 全 局 相 限 (全 选 /全 剖 不 选 ) 
履 误 : My SQL 夫 且 各 俯 妆 亿 贡 芝 旦 而 


INSERT 
DUFDATE 
回 DELETE 
FLE 


DROP 

CREATE TEMPORARY TABLES 
CREATEVEW 

口 ShowVvEW 

CREATE ROUTINE 

ALTER ROUTINE 

四 EXECUTE 


图 8-21 账户 (testuser'@'ocalhost) 只 用 于 从 本 机 连接 


您 已 添加 了 一 个 新 用 户 。 
SQL 查询 - 


CREATE USER lestuser@ % IDENTIFIED BY 


GRANT USAGE ON ”TO testuser@ % IDENTIFIED BY WITH MAX_QUERIES_PER_HOUR 0 


MAX_CONNECTIONS_PER_HOUR 0 MAX_UPDATES_PER_HOUR 0 MAX_USER_CONNECTIONS 0 


用 户 主机 ”密码 。 全 局 权限 四 。 投 权 
厂 root localhost 是 ALL FRIVIIEGES 是 
的。 testuser localhost 是 UsAGE 于 ”多 
企 _ 全 选 /全 部 不 选 
获 添加 新 用 户 


所 


三 化 “删除 选中 用 户 . 
(收回 用 户 的 所 有 激活 权限 ， 然 后 删除 用 户 。 ) 


图 8-23 ”删除 用 户 
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也 被 取消 ,此 用 户 再 次 试图 登录 时 将 会 失败 。 


8.4.3 权限 分 配 


1. MySQL 访问 权限 系统 

MySQL 权限 系统 保证 所 有 的 用 户 只 执行 允许 做 的 事情 。 当 用 户 连 接 MySQL 服务 
器 时 ,用 户 的 身份 由 他 从 那儿 连接 的 主机 和 指定 的 用 户 名 来 决定 。 连 接 后 发 出 请 求 后 , 系 
统 根据 用 户 的 身份 和 用 户 想 做 什么 来 授予 权限 。 

MySQL 存 取 控制 包含 两 个 阶段 : 

阶段 1 一 一 服务 器 检查 是 否 允 许 该 用 户 连 接 。 

阶段 ?一 一 假定 该 用 户 能 连接 ,服务 器 检查 他 发 出 的 每 个 请 求 。 看 他 是 否 有 足够 的 
权限 实施 它 。 例 如 ,如 果 这 个 用 户 从 数据 库 表 中 选择 (SELECT) 行 或 从 数据 库 删 除 表 , 那 
么 服务 器 会 确定 他 对 表 有 SELECT 权限 或 对 数据 库 有 DROP 权限 。 

如 果 连 接 时 用 户 的 权限 被 管理 员 更 改 了 ,这 些 更 改 不 一 定 立 即 对 发 出 的 下 一 个 语句 
生效 。 

服务 器 在 MySQL 系统 数据 库 的 授权 表 中 保存 权限 信息 。 当 MySQL 服务 器 启动 时 
将 这 些 表 的 内 容 读 和 人 内存。 

2. MySQL 提供 的 权限 

MySQL 提供 的 权限 的 如 图 8-24 所 示 。 


厂 全 局 权限 〈 全 选 /全 部 不 选 ) 
此 谨 : 朋 SQL 妈 忻 访 售 从 以 责 葡 星 开 


厂 SHOW DATABASES 
厂 IocK TABLES 

厂 REFERENCES 

厂 REPLICATION 
CLIENT 

厂 REPLICATION 


图 8-24 MySQL 提供 的 权限 


“全 局 权限 ?是 指 账户 对 MySQL 中 所 有 数据 库 与 数据 表 都 生效 的 权限 。MySQL 提 
供 的 权限 分 为 三 大 块 。 

1) 数据 管理 

SELECT、INSERT、UPDATE 和 DELETE 权限 允许 用 户 在 一 个 数据 库 现 有 的 表 上 
实施 操作 。 
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FILE 权限 给 予 用 户 用 LOAD DATA INFILE 和 SELECT ... INTO OUTFILE 语 
句 读 和 写 服务 器 上 的 文件 ,任何 被 授予 FILE 权限 的 用 户 都 能 读 或 写 MySQL 服务 器 能 
读 或 写 的 任何 文件 。FILE 权限 允许 用 户 在 MySQL 服务 器 具有 写 权限 的 目录 下 创建 新 
文件 ,不 能 覆盖 已 有 文件 。 

2) 数据 库 与 数据 表 的 结构 管理 

通过 CREATE 和 DROP 权限 ,可 以 创建 新 数据 库 和 表 , 或 删除 已 有 的 数据 库 和 表 。 

通过 ALTER 权限 ,可 以 使 用 ALTER TABLE 来 更 改 表 的 结构 和 重新 命名 表 。 

INDEX 权限 允许 用 户 创建 或 删除 索引 。INDEX 适用 已 有 表 。 

需要 CREATE ROUTINE 权限 来 创建 保存 的 程序 (函数 和 程序 ), ALTER 
ROUTINE 权限 来 更 改 和 删除 保存 的 程序 ,EXECUTE 来 执行 保存 的 程序 。 

3) MySQL 系统 管理 

GRANT 权限 允许 用 户 把 自己 拥有 的 那些 权限 授 给 其 他 的 用 户 。 

RELOAD 命令 告诉 服务 器 将 授权 表 重 新 读 入 内 存 。FLUSH-PRIVILEGES 是 
RELOAD 的 同义词 ,REFRESH 命令 清空 所 有 表 并 打开 并 关闭 记录 文件 ,其 他 FLUSH- 
XXX 命令 执行 类 似 REFRESH 的 功能 ,但 是 范围 更 有 限 . 并 且 在 某 些 情况 下 可 能 更 
好 用 。 

SHUTDOWN 命令 关 掉 服务 器 。 

PROCESSLIST 命令 显示 在 服务 器 内 执行 的 线程 的 信息 ( 即 其 他 账户 相关 的 客户 端 
执行 的 语句 )。KILL 命令 杀 死 服务 器 线程 。 

拥有 CREATE TEMPORARY TABLES 权限 便 可 以 使 用 CREATE TABLE 语句 中 
的 关键 字 TEMPORARY。 

拥有 LOCK TABLES 权限 便 可 以 直接 使 用 LOCK TABLES 语句 来 锁定 拥有 
SELECT 权限 的 表 , 包 括 使 用 写 锁定 ,可 以 防止 他 人 读 锁定 的 表 。 

拥有 REPLICATION CLIENT 权限 便 可 以 使 用 SHOW MASTER STATUS 和 
SHOW SLAVE STATUS。 

REPLICATION SLAVE 权限 应 授予 从 属 服务 器 所 使 用 的 将 当前 服务 器 连接 为 主 服 
务 器 的 账户 。 没 有 这 个 权限 ,从属 服务 器 不 能 发 出 对 主 服务 器 上 的 数据 库 所 发 出 的 更 新 

拥有 SHOW DATABASES 权限 便 允许 账户 使 用 SHOW DATABASE 语句 来 查看 
数据 库 名 。 没 有 该 权限 的 账户 只 能 看 到 他 们 具有 部 分 权限 的 数据 库 ,如 果 数 据 库 用 一 
skip-show-database 选项 启动 , 则 根本 不 能 使 用 这 些 语句 。 

总 的 说 来 ,只 将 权限 授予 给 需要 它们 的 那些 用 户 是 好 主意 。 

需要 注意 的 是 , 授 给 MySQL 数据 库 本 身 的 权限 能 用 来 改变 密码 和 其 他 访问 权限 信息 。 
密码 被 加 密 存 储 , 所 以 恶意 的 用 户 不 能 直接 读 取 明 文 密码 。 然 而 ,具有 user 表 Password 列 
写 访问 权限 的 用 户 可 以 更 改 账户 的 密码 ,并 可 以 用 该 账户 连接 MySQL 服务 器 。 

MySQL 权限 系统 无 法 做 到 : 

。 不 能 直接 指定 某 个 给 定 的 用 户 应 该 被 拒绝 访问 。 即 ,你 不 能 直接 指定 用 户 拒绝 连 

接 的 权限 。 
。 不 能 指定 用 户 有 权 创 建 或 删除 数据 库 中 的 表 、 但 不 能 创建 或 删除 数据 库 本 身 的 


187 


188 PHP+MySQL 项 目 实例 开发 


权限 。 
3. 使 用 phpMyAdmin 管理 权限 
添加 完 用 户 以 后 ,管理 界面 的 上 方 会 提示 “您 已 添加 了 一 个 新 用 户 ”, 我 们 现在 就 给 新 
建 的 用 户 添 加 权限 ,在 下 面 找到 “ 按 数据 库 指定 权限 ”选项 区 域 ,然后 单 击 “ 在 下 列 数 据 库 
添加 权限 ”下 拉 列 表 , 选 择 之 前 已 创建 的 数据 库 mydatabase, 如 图 8-25 所 示 。 


按 数 据 库 指定 权限 
数 括 库 权限 授权 ” 按 表 指定 权限 操作 
再 


在 下 列 数据 库 添加 权限 : | 使 用 文本 域 可 9 
使 用 文本 域 


mysql 
更 改 密码 phpmyadmin 
和 ~ 无 g 码 est 


图 8-25” 按 数据 库 指定 权限 


浏览 器 会 自动 转 和 人 按 已 选择 的 数据 配置 权限 的 页 面 ,如 图 8-26 所 示 。 选 择 “ 数 据 ” 选 
项 区 域 中 的 全 部 复 选 框 ,然后 选择 “结构 ”选项 区 域 中 的 前 七 个 复 选 框 ,其 他 一 律 不 选 , 然 
后 单 击 “ 执 行 ”按钮 ,这 样 就 配置 好 了 这 个 用 户 完全 管理 这 个 数据 库 的 权限 了 。 


多 用户 ‘testuser'@'1ocalhost - 数据 库 mydatabase : 编辑 权限 


三 按 数据 库 指 定 权 限 (全 选 /全 部 不 选 ) 
此 麻 ， My SQL 詹 怕 售 凉 佐久 下文 星 开 


区 INsERT 


VS UPDRTE 
5 DELETE 


8-26 ”为 用 户 配 置 权限 


在 这 里 我 们 只 是 要 给 这 个 用 户 管理 这 个 数据 库 的 全 部 权限 ,但 是 没有 其 他 数据 库 的 
管理 权限 ,所 以 在 配置 权限 的 时 候 一 定 要 注意 。 按 上 述 的 配置 方法 ,testuser 用 户 将 只 能 
从 本 机 登入 ,只 能 管理 mydatabase 这 个 数据 库 的 结构 和 数据 。 

此 外 .如果 在 创新 用 户 时 ,选择 “用 户 数据 库 ” 中 的 “给 以 用 户 名 _ 开 头 的 数据 库 
(username\_%) 授 予 所 有 权限 ”选项 ,新 建 的 用 户 就 可 以 自己 创建 以 “username_” 开 头 的 
数据 库 。 


第 8 章 SQL 与 MySQL 


8.5 项 目 训 练 


购物 类 网 站 产品 目录 的 数据 库 设计 


8.5.1 项 目 说 明 


产品 目录 可 以 说 是 购物 类 网 站 的 灵魂 ,购物 网 站 中 的 所 有 活动 都 涉及 产品 目录 。 产 
品目 录 包 含 购物 网 站 中 所 有 商品 的 信息 。 

产品 目录 的 结构 应 该 为 树 状 结构 , 树 状 结构 是 在 数据 库 中 查找 记录 的 算法 。 这 种 算 
法 的 每 个 元 素 为 一 个 结 点 。 结 点 可 以 分 为 许多 分 支 , 即 子 结 点 。 树 状 结构 中 的 所 有 记录 
都 存储 在 叶子 位 置 。 不 存在 任何 超越 记录 的 结构 。 这 就 好 像 一 棵 真正 的 树 一 样 ,有 主干 ， 
主干 上 连 着 许多 树枝 ,最 末端 是 叶子 。 叶 子 的 起 源 处 为 根 。 


8.5.2 设计 思路 


将 树 状 结构 与 购物 网 站 的 产品 目录 结构 进行 比较 。 在 一 般 的 购物 网 站 中 ,产品 分 为 
若干 个 大 类 ,每 类 下 面 又 分 为 子 类 ,并 且 每 个 商品 都 根据 它 的 分 类 进行 存储 。 除 父 类 外 ， 
每 个 种 类 都 在 某 类 或 其 他 类 的 下 一 级 。 在 此 基础 上 ,商品 可 以 被 视 为 树 状 结构 的 叶 结 点 ， 
并 且 各 类 为 子 结 点 。 例 如 ,“ 单 鞋 " 可 以 在 “ 女 鞋 "下 加 以 分 类 .“ 女 鞋 " 类 别 本 身 的 父 类 为 
“ 鞋 包 配 饰 ”。 

这 些 类 别 存储 在 数据 库 的 类 别 表 中 。 当 顾客 查找 特定 类 别 的 商品 时 ,可 以 用 产品 目 
录 的 搜索 选项 。 搜 索 结 构 很 大 程度 上 受到 数据 库 设计 的 影响 。 决 定数 据 库 效率 和 有 效 性 
的 主要 因素 是 在 数据 库 中 搜索 数据 是 否 容易 以 及 速度 如 何 。 在 设计 数据 库 结构 时 ,应 该 
以 快速 而 容易 地 获得 结果 为 宗旨 。 

为 此 ,该 数据 库 至 少 需要 两 张 数据 表 , 分 别 存 储 商 品 的 信息 表 (product 表 ) 以 商品 的 
分 类 表 (category 表 ) 。 


8.5.3 设计 过 程 


商品 表 (product 表 ) 的 字段 定义 应 该 包含 商品 ID .商品 名 称 、 商 品 描述 .商品 价格 以 
及 商品 所 属 分 类 ,如 表 8-9 所 示 。 

分 类 表 (category 表 ) 的 字段 定义 应 该 包含 分 类 ID、 分 类 名 称 、 分 类 描述 以 及 父 类 ,如 
表 8-10 所 示 。 


表 8-9 product 表 的 最 终 描述 。 其 中 product_id 表 8-10 category 表 的 最 终 描述 。 其 中 category_id 


将 被 标记 为 自动 递增 的 主键 将 被 标记 为 自动 递增 的 主键 
列 名 类 型 列 名 类 型 
productid | INT(10) UNSIGNED NOT NULL Ce 
和 人 Blas nainé VARCHAR(100) NOT NULL 
i FLOAT NOT NULL description TEXT NOT NULL 


category | INT(10) UNSIGNED NOT NULL parentcategory | INT(10) UNSIGNED NOT NULL 
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在 phpMyAdmin 中 将 上 述 数据 库 设 计 实 现 ,创建 product_catalog 数据 库 , 并 在 其 中 
创建 product 表 和 category 表 , 配 置 相关 字段 ,数据 字典 如 图 8-27 所 示 。 


图 服务 器 : localhost ， 数据 库 : product_catalog 


己 Open new phpMyAdmin window 


图 8-27 为 用 户 配置 权限 


本 章 小 结 


在 本 章 中 介绍 了 如 何 合理 进行 数据 库 设计 ,以 及 使 用 不 同方 式 访问 与 管理 MySQL。 
还 学 习 创建 表 、 填 充 它 们 以 及 运行 其 他 基本 查询 所 需 知道 的 所 有 SQL。 


重点 回顾 
1. MySQL 的 数据 类 型 。 
2. 如 何 操作 phpMyAdmin。 
3. 基本 SQL 语句 。 
4. 如 何 管理 MySQL 用 户 。 

本 章 实 训 
【 实 训 】 


设计 ”留言 板 ? 系 统 的 数据 库 , 其 中 至 少 包含 两 张 表 , 分 别 存储 留言 与 用 户 的 信息 , 自 
行 创建 新 的 数据 库 、 创 建新 表 、 设 计 字段 .添加 各 种 注释 ,学 习 区 分 不 同 的 情况 该 选择 的 数 
据 类 型 ,特别 注意 字符 编码 。 每 张 表 至 少 手动 输入 5 条 记录 ,测试 中 英文 记录 内 容 。 


第 9 章 使 用 PHP 和 MySQL 


在 网 站 项 目 中 ,网 站 的 数据 是 存放 在 数据 库 中 的 。PHP 十 MySQL 是 一 个 非常 好 的 
网 站 解决 方案 ,具有 非常 好 的 性 能 和 安全 性 。 在 PHP 网 站 中 ,程序 常常 进行 数据 库 访 
问 ,数据 库 访 问 是 通过 SQL 语句 完成 的 。 本 章 讲解 PHP 对 MySQL 数据 库 进行 连接 、 表 
单 使 用 .数据 查询 等 方面 的 内 容 。 本 章 中 使 用 的 数据 库 , 是 上 一 章 建立 的 mydatabase 数 
据 库 。 在 进行 本 章 学 习 之 前 ,需要 按照 上 一 章 的 内 容 建立 数据 库 。 


9.1 连接 MySQL 


在 PHP 中 连接 MySQL ,建立 数据 库 连 接 , 执 行 SQL 查询 ,最 后 通过 某 种 适合 Web 
应 用 程序 的 方法 来 处 理 查 询 结果 。 

在 访问 数据 库 以 前 需要 连接 数据 库 服务 器 。 而 一 个 MySQL 中 常常 有 多 个 数据 库 ， 
在 连接 成 功 以 后 ,需要 选择 一 个 数据 库 。 在 实际 开发 中 ,需要 将 数据 库 的 连接 信息 写成 一 
个 配置 文件 ,在 网 页 中 只 需要 包含 这 个 文件 即 可 完成 数据 库 的 连接 。 


9.1.1 连接 到 MySQL 


PHP 的 内 置 功能 支持 连接 MySQL ,能 够 建立 连接 .执行 查询 和 处 理 结 果 。PHP 中 
大 量 以 mysql_ 开 头 的 函数 都 有 这 种 神奇 的 能 力 ,可 以 完成 这 些 工 作 。 虽 然 这 些 函 数 很 有 
用 ,但 以 后 可 能 会 希望 Web 应 用 程序 能 使 用 各 种 数据 库 。 为 此 ,还 使 用 第 三 方 数据 库 抽 
象 层 及 抽象 类 ,如 PEAR::DB 或 ADODB。 如 果 使 用 MySQL 仅仅 是 为 了 某 个 特定 项 
目 , 那 么 mysql_ 系列 函数 就 足以 胜任 了 。 

首先 要 实际 地 建立 一 个 数据 库 连 接 。 在 发 送 查 询 和 收发 数据 时 ,可 以 利用 这 个 实际 
的 连接 与 数据 库 进 行 通信 。 为 此 需要 编写 一 段 PHP 代码 ,告诉 MySQL 相关 的 认证 信 
息 。 认 证 成 功 后 就 能 得 到 数据 库 连 接 。 

要 想 让 MySQL 服务 器 为 我 们 工作 ,建立 连接 MySQL 服务 器 的 通道 是 必 不 可 少 的 。 
在 PHP 中 ,连接 MySQL 是 一 件 很 容易 的 事情 .在 MySQL 服务 器 已 经 开启 的 情况 下 ,使 
用 mysql_connect() 函数 就 能 轻松 办 到 。 

MySQL 服务 器 的 连接 是 通过 mysql_connect 函数 实现 的 ,用 法 如 下 : 


$ dbc=mysql connect ($host, $username, $password); 
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这 个 函数 连接 MySQL 服务 器 以 后 ,会 返回 一 个 资源 类 型 。 函 数 参 数列 表 的 含义 如 
下 所 示 
。 $host 服务 器 主机 所 在 的 地 址 ,可 以 是 一 个 IP 地 址 或 者 域名 。 当 服务 器 是 
本 地 计算 机 时 ,主机 名 可 以 是 localhost 或 127.0.0. 1。 
访问 服务 器 的 用 户 名 。 
。 $ password 用 户 的 密码 。 
例如 ,用 户 连接 本 机 MySQL 服务 器 的 代码 如 下 : 


$dbc=mysql_connect ("localhost", "root", "root"); 

如 果 函 数 调 用 成 功 ,并 成 功 建立 了 连接 ,$ dbc 变量 将 成 为 后 续 数 据 库 交 互 的 一 个 参 
考点 。 用 于 处 理 MySQL 的 大 多 数 PHP 函数 都 可 以 把 它 作 为 一 个 可 选 的 参数 ,但 是 ,如 
果 省 略 了 它 , 这 些 函 数 也 会 自动 使 用 打开 的 连接 。 如 果 连 接 失 败 , 则 返回 FALSE。 


9.1.2 选择 数据 库 


一 个 MySQL 服务 器 中 ,常常 有 多 个 数据 库 。 连 接 MySQL 服务 器 以 后 ,选择 需要 使 
用 的 数据 库 。mysql_select_db() 函 数 用 于 选择 目标 数据 库 , 用 法 如 下 : 

mysql_select_db($ database name, $1ink identifier); 

mysql_select_db() 函 数 用 来 选择 MySQL 服务 器 中 的 数据 库 。 参 数 $ database_ 
name 是 数据 库 名称 的 字符 串 。$link_identifier 是 一 个 资源 型 变量 ,表示 连接 服务 器 时 
返回 的 资源 ,这 个 参数 是 可 选 参数 ,如 果 没 有 设置 , 则 表示 已 经 连接 的 服务 器 。 函 数 的 返 
回 值 是 一 个 布尔 值 ,如 果 成 功 , 则 返回 TRUE; 如 果 失 败 , 则 返回 FALSE。 

例如 ,选择 mydatabase 数据 库 , 代 码 如 下 : 


» $username 


$dbc=mysql_connect ("localhost", "root", "root"); 

mysql_select_db ("mydatabase",$ dbc)or die ("选择 数据 库 失 败 "); 

先 使 用 mysql_connect 函数 建立 数据 库 连接 ,然后 使 用 连接 成 功 的 $ dbc 参数 选择 数 
据 库 mydatabase, 若 失败 则 停止 PHP 程序 的 运行 ,并 提示 错误 信息 。mysql_select_db() 
函数 选择 一 个 数据 库 , 相 当 于 MySQL 命令 行 中 的 USE 命令 。 


9.1.3 关闭 数据 库 连接 


一 且 使 用 完 现 有 的 MySQL 连接 ,就 要 关闭 它 。mysql_close() 用 于 关闭 数据 库 连 接 ， 
用 法 如 下 : 


mysql close($1ink identifier) 


$ link_identifier 是 一 个 资源 型 变量 ,表示 连接 服务 器 时 返回 的 资源 ,这 个 参数 是 可 
选 参数 ,如 果 没 有 指定 ,默认 使 用 最 后 被 mysql_connect() 打 开 的 连接 。 

mysql_close( ) 函数 关闭 指定 的 连接 标识 所 关联 的 到 MySQL 服务 器 的 非 持久 连接 。 
如 果 没有 指定 $ link_identifier, 则 关闭 上 一 个 打开 的 连接 。 如 果 关 闭 成 功 , 则 返 


回 
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TRUE; 如 果 失 败 , 则 返回 FALSE。 例 如 : 


$ dbc=mysql_connect ("localhost", "root", "root"); 


mysql_close ($dbc); 


建立 $ conn 连接 后 ,关闭 该 连接 。 
mysql_close() 函数 不 是 必需 的 ,因为 在 脚本 的 末尾 ,PHP 会 自动 关闭 连接 ,但 是 使 用 
这 个 函数 确实 是 一 种 良好 的 编程 风格 。 


9.1.4 网 站 配置 文件 


在 PHP 网 站 中 ,几乎 所 有 的 文件 都 需要 连接 MySQL 服务 器 和 选择 数据 库 , 可 以 将 
这 些 功 能 写 在 一 个 文件 中 ,需要 访问 数据 库 的 网 页 只 需要 包含 这 个 文件 即 可 实现 
MySQL 数据 的 连接 功能 。 而 数据 库 的 连接 参数 ,常常 是 写 在 一 个 配置 文件 中 的 。 这 种 
模块 公用 的 方法 就 是 网 站 配置 文件 。 当 MySQL 服务 器 的 信息 发 生变 化 时 ,只 需要 更 改 
配置 文件 的 参数 即 可 。 

以 下 将 创建 一 个 针对 此 目的 的 特殊 文件 ,来 演示 连接 到 MySQL 的 过 程 。 需 要 
MySQL 连接 的 其 他 PHP 脚本 可 以 包含 这 个 文件 。 

在 此 之 前 ,有 必要 介绍 一 个 mysql_error() 函 数 。 

1. PHP 的 mysql_error( ) 函数 

错误 处 理 在 任何 脚本 中 都 很 重要 ,在 处 理 数据 库 时 ,这 是 个 大 问题 ,因为 发 生 错误 的 
可 能 性 会 显著 增加 。 打 印 出 查询 结果 ,然后 通过 MySQL 客户 端 运 行 它们 ,这 是 一 种 关键 
的 调试 技术 ,但 是 , 它 肯定 不 是 唯一 的 工具 。 

为 了 让 脚本 提供 关于 所 发 生 错误 的 详尽 报告 ,可 以 使 用 mysql_error() 函 数 。 它 会 返 


回 MySQL 与 前 一 个 数据 库 交 互 时 所 发 生 的 错误 人 f 


无 论 是 在 尝试 连接 到 MySQL 时 


出 错 , 在 选择 要 使 用 的 数据 库 时 发 生 错 误 ,还 是 在 运行 查询 时 发 生 错误 ,mysql_error() 画 
数 都 将 报告 任何 问题 ,就 像 在 MySQL 客户 端 中 运行 了 这 个 命令 一 样 。 在 数据 库 操 作 中 ， 
将 广泛 使 用 这 个 函数 。 

2. 连接 到 并 选择 数据 库 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 9-1) 。 


脚本 9-1 本 脚本 建立 了 一 条 到 达 MySQL 的 连接 ,并 选择 数据 库 。 


和 
2 
3 
4 
5 
6 
7 
8 


< ?php #Script 9-1 -mysql_connect.php 


Shostname= "localhost"; 
Sdatabase= "mydatabase"; 
$username= "username"7 


Spassword= "password"; 


$dbc=@mysql pconnect ($ hostname, $username, $password)OR die(' 连 接 数 据 库 失败 :' 
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- mysql_error ())7 
9 

10 @mysql select db ($ database)OR die(' 选 择 数据 库 失败 :' . mysql_error()) ?7 
11 ?> 


(2) 将 数据 库 主 机 、 用 户 名 、 密 码 和 数据 库 名 称 设置 为 变量 。 

$hostname= "localhost"; 

$ database= "mydatabase"; 

$username= "username"; 

$ password= "password"; 

把 这 些 设置 为 变量 比较 有 意义 ,这 样 就 能 够 把 配置 参数 与 使 用 它们 的 函数 隔离 开 ,但 
是 这 不 是 必需 的 。 

在 编写 脚本 时 ,可 以 把 这 些 值 更 改 为 处 理 数据 库 对 应 的 值 ,并 把 适当 的 权限 授予 名 为 
username 的 用 户 ,其 密码 为 password。 

(3) 连接 到 MySQL。 

$ dbc=@ mysql_pconnect ($ hostname, $username, $password)OR die(' 连 接 数 据 库 失败 :' . mysql_ 

error ()); 

如 果 成 功 连接 到 MySQL, 则 mysql_connect() 函 数 将 返回 对 应 于 打开 连接 的 资源 链 
接 。 这 个 链接 将 被 赋予 $ dbc 变量 ,用 来 与 数据 库 连接 ,在 使 用 其 他 MySQL 特有 的 函数 
时 ,这 个 连接 可 以 让 用 户 引用 。 

在 函数 调用 前 面 放置 了 一 个 错误 抑制 运算 符 (@), 可 以 防止 在 Web 浏览 器 中 显示 
PHP 错误 ,这 是 一 种 首选 的 做 法 ,因为 错误 将 由 OR die 子 句 处 理 。 

如 果 mysql_connect() 函 数 不 能 返回 有 效 的 资源 链接 ,那么 就 会 执行 语句 的 OR die 
〇 部 分 ,因为 OR 的 第 一 部 分 将 为 假 , 因 此 第 二 部 分 必须 为 真 。die() 函 数 会 终止 脚本 的 
执行 。 该 函数 还 可 以 取 一 个 指向 Web 浏览 器 的 字符 串 作 为 参数 。 在 这 种 情况 下 ,这 个 字 
符 串 是 “连接 数据 库 失 败 : ”的 字符 串 和 特定 的 MySQL 错误 的 组 合 。 在 开发 站 点 时 ,使 
用 这 个 错误 管理 系统 ,使 得 调试 容易 得 多 。 

(4) 选择 要 使 用 的 数据 库 , 并 关闭 PHP 页 面 。 


@mysql_select_db($database)OR die(" 选 择 数据 库 失败 :' . mysql_error());; 

2 

最 后 这 一 步 告诉 MySQL 每 个 查询 应 该 运行 在 什么 数据 库 上 。 如 果 选 择 数 据 库 失 
败 , 则 会 在 后 面 的 脚本 中 引发 问题 ,尽管 如 果 应 用 程序 使 用 了 多 个 数据 库 , 可 以 不 必 在 此 
选择 一 个 。 

同样 ,OR die() 构 造 用 于 处 理 任何 MySQL 问题 ,@ 会 抑制 原来 的 PHP 错误 。 

(5) 将 文件 另存 为 mysql_connect. php。 

因为 这 个 文件 包含 必须 保持 私有 的 数据 库 访 问 信息 ,所 以 将 使 用 . php 后 缀 名 。 利 
用 . php 后 缀 名 ,有 恶意 的 用 户 即 使 在 其 Web 浏览 器 中 运行 了 这 个 脚本 ,也 不 会 看 到 页 面 
的 实际 内 容 。 


第 9 章 ”使 用 PHP 和 MySQL 


(6) 将 这 个 文件 上 传 到 服务 器 ,建议 把 它 存 放 在 Web 文档 根 目录 的 外 面 。 

因为 这 个 文件 包含 敏感 的 MySQL 访问 信息 ,所 以 应 该 安全 地 存储 它 。 如 果 可 以 的 
话 , 就 把 它 放 在 Web 目录 的 上 一 级 目录 中 ,或 者 把 它 放 在 Web 目录 的 外 面 ,这 样 就 不 能 
从 Web 浏览 器 访问 该 文件 了 。 

(7) 在 Web 文档 根 目 录 中 临时 存放 一 个 这 个 脚本 的 副本 ,并 在 Web 浏览 器 中 运行 
这 个 脚本 (参见 图 9-1 和 图 9-2) 。 
E a 
(WS hepyfocalhosbr PD + OX | iocaltost 


9-1 连接 成 功 将 是 空白 页 面 


(HS @ htpy/localhosVbe PD ~ © X||localhost 
连接 数据 库 失败 ，Access denied for user 


“testuser' @ 1ocalhost” (using password: YES) 


图 9-2 连接 失败 将 会 提示 错误 信息 ,并 中 止 脚本 


为 了 测试 这 个 脚本 ,把 它 的 一 个 副本 放 在 Web 目录 中 ,使 得 可 以 从 Web 浏览 器 中 访 
问 它 。 如 果 脚 本 正确 地 工作 ,其 结果 应 该 是 一 个 空白 页 面 ,如 图 9-1 所 示 , 因 为 该 脚本 没 
有 生成 任何 的 HTML。 如 果 看 到 Access denied 或 者 类 似 的 消息 ,如 图 9-2 所 示 ,这 意味 
着 用 户 名 、 密 码 和 主机 的 组 合 不 具有 访问 特定 数据 库 的 权限 ,此 外 ,还 要 确认 MySQL 正 
在 运行 。 

(8) 测试 成 功 后 ,从 Web 目录 中 删除 临时 的 副本 。 

一 旦 编写 了 这 个 脚本 ,就 可 以 轻松 地 更 改变 量 定义 的 那 几 行 代码 ,可 以 把 这 个 脚本 用 
于 其 他 项 目 。 另 外 ,如 果 接 收 到 一 个 错误 .声称 mysql_connect() 是 一 个 未 定义 的 函数 ,这 
意味 着 没有 包含 MySQL 支持 来 编译 PHP。 如 果 需 要 在 同一 脚本 中 连接 到 多 个 数据 库 
服务 器 ,可 以 多 次 运用 mysql_connect() 函 数 , 并 把 返回 的 结果 赋予 不 同 的 PHP 变量 。 

图 9-3 显示 了 如 果 在 mysql_connect() 前 面 没有 使 用 错误 抑制 运算 符 @ 符 号 ,并 且 发 
生 一 个 错误 时 ,会 出 现 什 么 情况 。 

3. 修改 模板 

本 章 的 所 有 页 面 都 是 同一 个 Web 应 用 程序 的 一 部 分 ,所 以 使 用 公共 模板 系统 是 有 必 
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> 多 htpy/localhosybc PD - OX 


Warning: mysql_pconnect() [function. mysql- 
peonnect]: Access denied for user 

“username’ @ localhost” (using password: YES) in 
C:\AppServ\www\book\ch09 


\includes\mysql_connect. php on line 8 
连接 数据 + Access denied for user 
“username’ @ localhost’ (using password: YES) 


9-3 ”如果 没 有 使 用 @ ,将 同时 看 到 PHP 错误 和 自 定义 的 OR die() 错 误 


要 的 。 在 这 里 将 不 会 从 头 开 始 创 建 一 个 新 模板 ,而 将 再 次 使 用 第 7 章 的 布局 ,并 且 只 对 头 
文件 中 的 导航 链接 做 了 微小 的 修改 。 

(1) 在 文本 编辑 器 中 打开 header. html( 参 见 脚本 7-2) 。 

(2) 更 改 链接 导航 (参见 脚本 9-2) 。 


脚本 9-2 用 新 的 导航 链接 修改 了 站 点 的 头 文件 。 


1 <!DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.0org/TR/xhtmll/DTD/xhtmll- transitional .dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http-equiv="content- type" content="text/html; charset=gb2312" /> 
6 <title><?php echo $page title;?></title> 

7 <linkhref="includes/layout.css" rel="stylesheet" type="text/css" /> 

8 </head> 

和 a 


10 <body> 
11 <div id="wrapper"> 
12 <div id= "content"> 


13 <div id= "nav"> 

14 <h3> 请 选择 < /h3> 

15 <ul> 

16 <1i class= "navtop"><a href= "index.php" title= "首页 "> 首页 </a></1i> 
17 <1i><a href= "register.php" title= "注册 "> 注册 </a></1i> 

18 <1i><a href= "view_users.php" title= "查看 用 户 "> 查 看 用 户 </a></1i> 
19 <1i><a href= "password.php" title= "修改 密码 "> 修改 密码 </a></1i> 

20 </ul> 

21 </div> 

22 <!--Script 9-2 -header.html 一 一 > 


本 章 的 所 有 示例 都 将 涉及 注册 、 查 看 用 户 和 修改 密码 这 些 页 面 。 
(3) 将 文件 另存 为 header. html。 
(4) 将 新 的 头 文件 上 传 到 Web 服务 器 ,将 其 与 footer. html 和 layout. css 一 起 存放 
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在 includes 文件 夹 中 。 在 Web 浏览 器 中 运行 index. php, 测 试 新 的 头 文件 ,如 图 9-4 
所 示 。 


一 一 
C33 OIG hpi/ocalhosVboc P - OX || GE may ! x 从 家 次 


9-4 动态 生成 的 主页 


9.2 执行 简单 查询 


9.2.1 执行 SQL 操作 


一 旦 成 功 连接 到 并 选择 一 个 数据 库 , 就 可 以 开始 执行 SQL 操作 。 在 PHP 中 ,使 用 
mysql_query() 函 数 来 向 MySQL 服务 器 发 送 各 种 不 同 的 SQL 语句 ,例如 插入 、 更 新 、 删 
除 和 查询 等 操作 。mysql_query() 函数 的 用 法 如 下 : 


$result=mysql query ($ query); 


其 中 参数 $ query 必须 是 一 个 正确 的 SQL 语句 。 

INSERT UPDATE 及 DELETE 这 样 简单 的 查询 ,都 不 会 返回 结果 ,对 于 这 样 简单 
的 SQL 操作 ,mysql_query() 函数 的 返回 值 $ result 将 是 TRUE 或 者 FALSE, 这 取决 于 
查询 是 否 执 行 成 功 。 对 于 会 返回 记录 的 复杂 查询 ,例如 ,SELECT、SHOW、EXPLAIN 和 
DESCRIBE 操作 ,如 果 查 询 有 效 . 则 $result 变量 将 是 一 个 指向 查询 结果 的 资源 链接 ;如 
果 查 询 无 效 , 则 $result 变量 将 为 FALSE。 


9.2.2 ”MySQL 的 字符 集 


1. 字符 集 
在 PHP 与 MySQL 数据 库 交互 的 过 程 中 :经常 出 现 乱码 的 问题 ,这 就 是 MySQL 字 
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符 集 的 问题 。 字 符 集 是 一 组 符号 以 及 与 它们 对 应 的 编码 ;校对 规则 是 一 组 规则 ,规定 了 字 
符 之 间 如 何 比 较 。 每 一 个 字符 集 都 对 应 着 一 组 (至 少 一 个 ) 校 对 规则 ,而 每 一 个 校对 规则 
对 应 唯一 一 个 字符 集 ,通常 它们 两 个 需要 成 对 出 现 , 已 完成 数据 库 里 的 相关 操作 ,比如 排 
序 , 字 符 串 连接 等 操作 。 

我 们 需要 了 解 哪些 字符 集 和 校对 规则 是 可 用 的 ,怎样 改变 默认 值 ,以 及 它们 怎样 影响 
字符 操作 符 和 字符 串 函 数 的 行为 。 

MySQL 对 于 常规 的 字符 集 能 够 做 以 下 这 些 事情 : 

。 使 用 多 种 字符 集 来 存储 字符 串 。 

。 使 用 多 种 校对 规则 来 比较 字符 串 。 

。 在 同一 台 服 务 器 ,同一 个 数据 库 或 甚至 在 同一 个 表 中 使 用 不 同 字符 集 或 校对 规则 

来 混合 字符 串 。 

。 允许 定义 任何 级 别 的 字符 集 和 校对 规则 。 

在 这 些 方面 ,MySQL 比 其 他 大 多 数 数据 库 管理 系统 超前 许多 。MySQL 的 字符 集 可 
以 使 用 在 四 个 级 别 上 : 服务 器 数据库、 数据 表 和 列 , 而 且 对 于 各 个 存储 引擎 都 支持 字符 
集 的 使 用 。 

在 上 述 四 个 级 别 上 ,对 字符 集 和 校对 规则 都 有 默认 的 设置 ,服务 器 层 的 默认 为 latin1 
和 latin1_swedish_ci。 在 创建 各 个 层次 的 实体 时 都 有 相应 的 子 句 或 者 候选 项 可 以 使 用 ， 
以 显 式 地 声明 各 个 实体 将 要 使 用 的 字符 集 和 校对 规则 。 

例如 ,在 第 8 章 使 用 phpMyAdmin 创建 数据 库 时 ,就 可 选择 gb2312_Chinese_ci 作为 
数据 库 的 默认 字符 集 编码 。 

2. 字符 集 的 系统 变量 

MySQL 提供 了 以 下 几 个 设置 字符 集 的 系统 变量 : 
客户 端 字符 集 。 
客户 端 与 服务 器 端 连接 采用 的 字符 集 。 

。 character_set_results SELECT 一 一 查询 返回 数据 的 字符 集 。 
数据 库 采 用 的 字符 集 。 

乱码 问题 一 般 是 由 于 以 上 几 个 变量 设置 错误 造成 的 ,所 以 只 要 理解 这 几 个 变量 ,就 可 
以 与 乱码 告别 了 。 

使 用 上 述 变 量 ,要 理解 这 个 核心 思想 : character _set _client 和 character _set _ 
connection 这 两 个 变量 保证 要 与 character_set_database 编码 的 一 致 ,而 character_set_ 
results 则 保证 SELECT 返回 的 结果 与 程序 的 编码 一 致 。 

3. SET NAMES 语句 

SET 语句 用 于 设置 不 同类 型 的 变量 。 这 些 变量 会 影响 服务 器 或 客户 端的 操作 。 
SET 可 以 用 于 向 用 户 变量 或 系统 变量 赋值 。 

其 中 SET NAMES 子 句 用 于 把 三 个 系统 变量 character_set_client、 character_set_ 
connection 和 character_set_results 设置 为 给 定 的 字符 集 。 

一 般 情 况 下 , 当 数据 库 与 数据 库 表 的 字符 集 为 utf8 时 ,再 在 程序 里 设置 set names 


。 character_set_client 


。 character_set_connection 


。 character_set_database 
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mutf8 俞 令 , 这 样 就 能 保证 无 乱码 了 ,但 是 ,这 里 还 要 注意 character_set_results 变量 的 值 ， 
character_set_results 的 字符 值 是 用 来 显示 返回 给 用 户 的 编码 的 。 

例如 ,数据 库 (character_ set_ database) 用 的 是 utf8 的 字符 集 , 那 么 就 要 保证 
character_set_client 及 character_set_connection 也 是 utf8 的 字符 集 。 而 程序 也 许 采用 
的 并 不 是 utf8 ,比如 程序 用 的 是 gb2312 ,那么 若 把 character_set_results 也 设置 为 utf8， 
就 会 出 现 乱 码 问题 。 此 时 应 该 把 character_set_results 设置 为 gb2312。 这 样 就 能 保证 数 
据 库 返回 的 结果 与 程序 的 编码 一 致 。 

在 本 书 中 ,所 有 的 页 面 使 用 的 网 页 编码 指定 为 gb2312, HTML 头 部 声明 中 有 如 下 


语句 : 


<meta http- equiv= "content-type”content= "text/html; charset=gb2312”/> 
因此 ,通过 PHP 页 面 与 MySQL 数据 库 交 互 时 ,必须 要 统一 字符 集 编码 。 为 避免 出 现 乱 
码 , 强 烈 建 议 在 每 次 执行 mysql_query() 函数 操作 数据 库 之 前 , 先 转 换 字符 集 编码 ,使 用 
如 下 代码 : 


mysql_query ("SET NAMES 'gb2312'"); 

通过 这 样 的 设 定 , 不 管 是 用 phpMyAdmin 管理 MySQL 数据 库 , 还 是 通过 PHP 页 面 
操作 数据 库 的 数据 ,中 英文 都 会 正常 存储 和 显示 。 

9.2.3 插 人 操作 


为 了 演示 这 个 过 程 ,我 们 将 编写 一 个 注册 脚本 , 它 类 似 第 6 章 的 脚本 6-6, 这 里 简化 至 
只 需 用 户 填 写 必 填 选 项 ,并 将 用 户 填 写 的 信息 输入 到 mydatabase 数据 库 的 user 表 中 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 脚本 (参见 脚本 9-3) 。 


脚本 9-3 本 脚本 向 数据 库 中 添加 一 条 记录 。 


< ?php #Script 9- 3 - register.php 
$page title=' 注 册 '; 
include ('./includes/header.html'); 


if(isset($_POST["Submit"])){ 

$errors=array(); 

if (empty ($_PosT[ 'name'])) { 
$errors[]=' 您 忘记 输入 用 户 名 ."'; 

} 

if (empty($_PosT['email'])){ 
$errors[]= ' 您 忘记 输入 您 的 EMAIL 地址 .7 

要 

if(lempty($_POsT['passwordl'])){ 
if($_POST["passwordl'] !=$_POST['password2']){ 

serrors[]= ' 两 次 输入 密码 不 同 .7 
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} 
} else { 
$errors[]= ' 您 忘记 输入 密码 .7 


if (empty ($errors)){ 
require_once("./includes/mysql_connect .php"); 
$sql="INSERT INTO 'users' ('user_id', 'name', 'password', 'email', 
"registration_date') 
VALUES (NULL, '{$_POST['name']}', SHA1('{$_POST['password1']}"), 
"{$_POST['email']}', NOW());"; 
mysql_query ("SET NAMES 'gb2312'"); 
$result=@mysql_query($ sq1); 


if($ result){ 
$body=" 谢 谢 您 注册 我 们 的 站 点 !'\n 您 的 登录 密码 为 :{$_POST["password1']}"; 
mail ($_POST['email'],，' 谢 谢 您 的 注册 !'，$body,'From: shiying@ zdxy.cn 
5 
echo '<hl id= "mainhead"> 注 册 成 功 !< /hl>'; 
echo "<p> 谢 谢 您 注册 我 们 的 站 点 !\n 您 的 登录 密码 为 :{5_POST['password1']} 
</p>"; 
echo '<p> 您 已 注册 成 功 , 确 认 邮 件 已 发 送 至 您 的 邮箱 !< /p><p>< /p>'; 
include ('./includes/footer .htm]l'); 
exit (); 

J}else{ 
echo '<hl id= "mainhead"> 系 统 错误 !< /hl> 
<p class= "error"> 很 抱 款 您 暂时 无 法 注册 。<br />'; 
echo '<p>"' . mysql error(); 
include ('./includes/footer.html'); 


exit (); 


mysql_close(); 
} else { 
echo '<hl id= "mainhead"> 错 误 !< /hl> 
<p class="error"> 出 现 以 下 错误 :<br /> 
foreach ($ errors as $msg) { 
echo " - $msg<br /> \n™; 
} 
echo '< /p><p> 请 重 填 :< /p><p><br /></p>'; 


?> 
<h2> 注 册 :< /h2> 
< form action= "register.php" method- "post"> 
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56 <table> 

EY <tr> 

58 <td> 用 户 名 : < /td> 

59 <td><input type= "text" name= "name" size="20" maxlength="40" value="< ?php 
if(isset($_POST['name']))echo 5_POST['name']; ?>" /></td> 

60 </tr> 

61 <tr> 

62 <td>Email 地 址 :< /td> 

63 <td>< input type= "text" name="email" size="20" maxlength="40" value="<?php if 
(isset($_POST["email"]))echo $_PoOST['email']; ?>" /></td> 

64 </tr> 

65 <tr> 

66 <td> 密 码 :< /td> 

67 <td><input type= "password" name= "passwordl" size= "10" maxlength= "20" /></ 
td> 

68 </tr> 

69 <tr> 

70 <td> 确 认 密码 : < /td> 

71 <td>< input type= "password" name= "password2" size= "10"” maxlength= "20"” /></ 
td> 

72 </tr> 

73 <tr> 

74 <td colspan= "2"><div align="center"> 

75 < input type= "submit" name= "Submit" value= "注册 "”/> 

76 </div>< /td> 

77 < /tr> 

78 </table> 

79 </form> 

80 <?php 

81 include('./includes/footer.html'); 

82 ?> 


(2) 设置 $ page_title 变量 ,并 且 包 含 HTML 头 文件 。 


$page title= ' 注 册 '; 
include('./includes/header.html'); 


(3) 创建 提交 条 件 语 句 , 并 初始 化 $ errors 数组 。 


if(isset($_POST['Submit"])){ 
$errors=array(); 
这 个 脚本 将 同时 显示 和 处 理 HTML 表单 。 这 个 条 件 语句 将 检查 是 否 处 理 表单 。 对 
$ errors 变量 进行 初始 化 ,以 使 得 后 面 构 建 它 时 不 会 产生 警告 。 
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(4) 验证 用 户 名 。 

证 (empty(5$_POST['name'])){ 
S$errors[]= ' 您 忘记 输入 用 户 名 ."; 

} 


empty() 函数 用 作 确 保 文件 框 不 空 的 方式 。 如 果 名 字 文 本 框 为 空 ,就 会 添加 一 条 出 
错 信息 到 $ errors 数组 中 。 
(5) 验证 电子 邮件 地 址 和 密码 。 
if(empty($5_POST['email'])){ 
5$errors[]= ' 您 忘记 输入 您 的 EMAIL 地 址 .7 
} 
if(!empty($_POST['password1'])){ 
if($_POST['password1'] !=$_POST['password2']){ 
$errors[]=' 两 次 输入 密码 不 同 ."; 
} else { 
$errors[]= ' 您 忘记 输入 密码 ."; 
} 


为 了 验证 密码 ,需要 检查 passwordl 输入 框 的 值 ,然后 确认 passwordl 与 password2 
的 值 是 否 相 等 。 
(6) 检查 用 户 是 否 正确 注册 。 


if (empty ($ errors)){ 


如 果 提 交 的 数据 通过 了 所 有 的 条 件 , 这 个 条 件 将 为 TRUE. 并 且 会 安全 地 执行 后 续 
操作 。 如 果 不 是 这 样 ,那么 应 该 打印 合适 的 出 错 信息 ,并 给 用 户 提供 另外 一 次 注册 的 
机 会 。 

(7) 将 用 户 添加 到 数据 库 中 。 


require once("./includes/mysql_connect.php"); 

$ sql= "INSERT INTO 'users ' ('user_id', 'name', 'password', 'email', 'registration date') 
VALUES (NULL , '{$_POST['name']}', SHAl1('{$_POST['password1']}"'), '{$_POST['email']}', 
NOW()) 2"; 

mysql_query ("SET NAMES 'gb2312'"); 

$result=mysql query($ sql); 


这 段 代码 的 第 一 行将 把 mysql_connect. php 文件 的 内 容 插入 到 这 个 脚本 中 ,从 而 创 
建 一 条 到 达 MySQL 的 连接 ,并 选择 数据 库 。 

这 个 查询 本 身 类 似 于 第 8 章 演示 的 SQL 操作 。SHA1() 函 数 用 于 对 密码 进行 加 密 ， 
NOWO 〇 函数 用 于 将 注册 日 期 设置 为 当前 时 间 。 在 这 里 需要 注意 的 是 各 个 $_POST 变量 
使 用 的 语法 , 当 数 组 使 用 字符 串 作为 它 的 键 时 ,应 把 数组 名 和 键 包装 在 花 括号 中 。 

在 运行 mysql_query() 函 数 , 先 统一 编码 为 gb2312 .避免 出 现 乱码 。 
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在 将 这 个 查询 赋予 变量 $ sql 之 后 ,就 会 通过 mysql_query() 函 数 运 行 , 它 会 把 SQL 
命令 发 送 到 MySQL 数据 库 。 就 像 在 mysql_connect. php 脚本 中 一 样 ,mysql_query() 调 
用 前 面 放置 了 一 个 @ 符 号 ,避免 错误 信息 直接 输出 在 页 面 中 。 如 果 发 生 问题 ,就 会 在 下 一 
步 中 更 直接 地 处 理 。 

(8) 报告 注册 成 功 。 


if($ result){ 
$body= "谢谢 您 注册 我 们 的 站 点 !\n 您 的 登录 密码 为 :{$_POST['password1']}"; 
mail ($_POST['email'],' 谢 谢 您 的 注册 !'，$body, 'From: shiying@ zdxy.cn'); 
echo '<hl id= "mainhead"> 注 册 成 功 !< /hl>'; 
echo "<p> 谢 谢 您 注册 我 们 的 站 点 !\n 您 的 登录 密码 为 :{5_POST['password1']}< /p>"; 
echo '<p> 您 已 注册 成 功 ,确认 邮件 已 发 送 至 您 的 邮箱 !< /p><p>< /p> '; 


include('./includes/footer.html'); 
exit (); 


} 


$result 变量 被 赋予 mysql_query() 函数 返回 的 值 , 它 可 以 用 在 条 件 语句 中 ,以 指示 
查询 的 成 功 操作 。 在 这 个 例子 中 ,也 可 以 通过 编写 如 下 条 件 语 句 , 来 节省 一 行 代码 : 


if (@mysql_query($ sql)){ 


如 果 $$result 为 TRUE, 就 会 显示 消息 ,包含 脚注 ,并 使 用 exit() 中 止 脚本 。 如 果 没 
有 在 这 里 包括 脚注 并 退出 脚本 ,就 会 再 次 显示 注册 表单 。 也 可 以 发 送 一 封 关于 成 功 注 册 
的 电子 邮件 。 

如 果 $result 为 FALSE, 就 会 打印 出 错 信 息 。 出 于 调试 的 目的 ,出 错 信息 将 通过 
mysql_error() 函 数 返回 MySQL 发 出 的 错误 。 同 样 会 包括 脚注 并 中 止 页 面 的 运行 ,以 使 
得 不 会 重新 显示 表单 。 

(9) 关闭 数据 库 连接 。 


mysql_close(); 


这 不 是 必需 的 ,但 它 是 一 个 良好 的 编程 习惯 。 
(10) 打印 出 任何 出 错 信息 ,并 关闭 提交 条 件 语 句 。 


} else { 

echo '<hl id= "mainhead"> 错 误 !< /hl> 

<P class= "error"> 出 现 以 下 错误 :<br /> '; 

foreach ($ errors as $msg) { 

echo " -$msg<br /> \n"; 

} 

echo '< /p><p> 请 重 填 :< /p><p><br /></p>'; 
} 
} 


如 果 有 任何 错误 ,就 会 调用 else 子 句 。 在 这 里 ,所 有 的 错误 都 用 foreach 循环 显示 。 
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最 后 的 右 花 括号 关闭 了 主 提交 条 件 语句 。 
(11) 关闭 PHP 代码 ,开始 HTML 表单 。 


<h2> 注 册 :< /h2> 
< form action= "register.php" method= "post"> 
<table width= "299" height="151"> 
<tr> 
<td> 用 户 名 : < /td> 
<td><input type= "text" name= "name" size="20" maxlength="40" value="<? php if 
(isset ($_POsT['name']))echo $_POsT['name']; ?>" /></td> 
</tr> 
<tr> 
<td> Email 地 址 :< /td> 
<td><input type= "text" name="email" size="20" maxlength="40" value="<?php if 
(isset ($_POST['email']))echo $_POST['email']; ?>" /></td> 
</tr> 


同样 ,为 了 让 表单 具有 黏 性 ,在 input 标签 的 value 属性 中 添加 PHP 代码。 

强烈 建议 为 表单 输入 框 使 用 与 数据 库 中 对 应 的 字段 相同 的 名 称 , 这 些 字段 中 存储 了 
表单 输入 框 的 值 。 此 外 ,应 该 把 表单 中 的 最 大 输入 框 长 度 设置 为 等 于 数据 库 中 的 最 大 列 
长 度 。 这 两 个 习惯 都 有 助 于 减少 错误 。 对 于 密码 输入 框 ,无 须 遵守 这 个 最 大 长 度 建议 , 因 
为 它们 将 会 用 SHA10 〇 函数 ,总 是 会 创建 一 个 长 度 为 40 个 字符 的 字符 串 。 

(12) 完成 HTML 表单 ,使 用 HTML 脚注 完成 页 面 。 


< /form> 

<?php 

include ('./includes/footer.html'); 
?> 


(13) 将 文件 另存 为 register. php, 上 传 到 Web 服务 器 上 与 index. php 相同 的 目录 
中 ,通过 在 Web 浏览 器 中 运行 这 个 脚本 来 测试 文件 (参见 图 9-5、 图 9-6、 图 9-7 和 图 9-8) 。 


Oe w/o*ess “CE 
| : 


图 9-5 注册 表单 
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OB cx 二 x 国 | 分 妆 冤 


注册 成 功 ! 


谢谢 您 注册 我 们 的 站 点 ! 您 的 登录 密码 为 : 12345678 
您 已 注册 成 功 ， 确认 邮件 已 发 送 至 您 的 邮箱 : 


© Copyright 2014 


图 9-6 如果 能 在 数据 库 中 注册 ,就 会 显示 成 功 的 信息 


系统 错误 ! 


很 抱 折 您 暂时 无 法 注册 。 
Unknown coumn 'd' in ‘fekd lst” 


© Copyright 2014 


图 9-7 由 查询 引发 的 MySQL 错误 
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【¢ OE x 国 分 妆 次 


© Copyright 2014 


图 9-8 表单 输入 有 误 及 黏 性 表单 


如 果 能 够 在 数据 库 上 执行 查询 ,mysql_query() 函数 就 会 返回 TRUE。 这 不 一 定 代 表 
查询 的 结果 就 是 我 们 所 期 待 的 。 所 以 在 运行 脚本 后 ,通过 使 用 phpMyAdmin 来 查看 
MySQL 数据 表 的 内 容 , 总 是 可 以 再 次 确保 操作 成 功 。 


9.3 检索 查询 结果 


9.3.1 处 理 查 询 结 果 数 组 


1. mysql_fetch_array() 函数 

在 前 一 节 中 ,可 以 把 简单 的 查询 定义 为 INSERT、UPDATE 或 DELETE 开头 的 一 个 
SQL 查询 。 所 有 这 3 种 查询 的 一 个 共同 点 是 : 它们 不 返回 任何 数据 ,而 只 返回 一 个 成 功 
或 失败 执行 的 指示 。 与 之 相反 ,SELECT 查询 会 生成 必须 由 其 他 PHP 函数 处 理 的 结果 ， 
因为 它 会 返回 几 行 记录 。 

处 理 SELECT 查询 结果 的 主要 工具 是 mysql_fetch_array() 函 数 , 它 带 有 一 个 查询 结 
果 变 量 ,也 就 是 前 面 定 义 的 $ result, 并 以 数组 格式 一 次 返回 一 行 数据 。 可 以 在 一 个 循环 
中 使 用 这 个 函数 ,只 要 有 更 多 的 行 , 循 环 就 会 持续 访问 返回 的 每 一 行 。 

mysql_fetch_array() 函数 从 结果 集中 取得 一 行 作为 联合 数组 .或 索引 数组 ,或 二 者 
兼 有 。 返 回 根据 从 结果 集 取得 的 行 生 成 的 数组 ,如 果 没 有 更 多 行 , 则 返回 FALSE。 

语法 如 下 


mysql_fetch array (data,array type) 
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从 查询 读 取 每 条 记录 的 基本 格式 如 下 : 
while($ row=mysql fetch array ($ result)){ 


// 处 理 $ row 
} 


mysql_fetch_array() 函数 带 有 一 个 可 选 的 参数 array_type, 用 于 指定 返回 的 数组 的 
类 型 : 联合 数组 、 索 引 数 组 ,或 者 这 两 者 同时 返回 。 联 合 数 组 允许 通过 名 称 引用 列 值 , 而 
索引 数组 则 要 求 只 使 用 数字 ,返回 的 第 一 列 , 从 0 开始 。 每 个 参数 都 是 通过 表 9-1 中 列 出 
的 常量 定义 的 。 
表 9-1 mysql_fetch_array() 函 数 的 可 选 参数 


常 量 示 例 
MYSQL_ASSOC $ row['name’] 
MYSQL_NUM $ row[1] 
MYSQL_BOTH $row[1] 或 $row[name] 


可 以 把 如 表 9-1 所 示 的 常量 之 一 作为 一 个 可 选 参数 添加 到 mysql_fetch_array() 函 
数 中 ,指示 如 何 访问 返回 的 值 。 该 函数 的 默认 设置 参数 为 MYSQL_BOTH。 与 其 他 选项 
相 比 ,MYSQL_NUM 设置 要 快 那么 一 点 点 ,并 且 使 用 更 少 的 内 存 。 与 之 相反 , MYSQL_ 
ASSOC 更 明确 ,即使 表 结 构 或 查询 发 生变 化 , 它 也 会 继续 工作 。 

需要 注意 的 是 ,mysql_fetch_array() 函 数 返回 的 字段 名 是 区 分 大 小 写 的 。 

2. 释放 内 存 

使 用 mysql_fetch_array() 函数 时 ,可 以 采取 的 一 个 可 选 的 步骤 是 : 一 旦 使 用 查询 结 
果 信 息 完成 了 工作 , 即 可 释放 这 些 信息 。 方 法 如 下 : 


mysql_free result ($result); 


参数 $ result 是 必需 的 ,是 要 释放 的 结果 标识 符 。 该 结果 标识 符 是 从 mysql_query() 
返回 的 结果 。 
这 一 行 会 消除 $ result 所 占用 的 系统 开销 。 这 个 步骤 是 可 选 的 ,mysql_free_result() 
wwe eaexwe ww wii 在 脚本 结束 后 所 有 关联 的 
内 存 都 会 被 自动 释放 。 就 像 使 用 mysql_close() 一 样 ,这 是 一 种 良好 的 编程 风格 。 


9.3.2 检索 查询 结果 


为 了 演示 如 何 处 理 查询 返回 的 结果 ,将 创建 一 个 脚本 ,用 户 查看 当前 注册 的 所 有 用 户 。 
(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 脚本 (参见 脚本 9-4) 。 


脚本 9-4 在 数据 库 上 运行 一 个 静态 查询 ,并 打印 出 返回 的 所 有 行 。 


1 <?php #Script 9-4 -view users.php 
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2 $page title=' 查 看 当前 所 有 用 户 '; 
include (' ./includes/header.html '); 
echo '<hl id= "mainhead"> 注 册 用 户 < /hl> '; 


require_once("./includes/mysql_connect .php"); 

$ sql= "SELECT name, DATE, FORMAT (registration date, '%Y 年 $m 月 %$d 日 ')RS dr FROM users 
ORDER BY registration_date ASC"; 

8 mysql_query("SET NAMES 'gb2312'"); 

9 $result=@mysql_query($ sql); 


10 

11 if($result){ 

12 echo '<table align= "center" cellspacing="0" cellpadding="5"> 

13 <tr><td><b> 姓 名 < /b>< /td><td><b> 注 册 时 间 < /b>< /td>< /tr>'; 
14 

15 while($ row=mysql_fetch array($ result, MYSQL ASSOC)){ 

16 echo '<tr><td 中 ' . $row['name'] . '</tdp<t 中 ! . $row['dr'] . '</tob</tr>'; 
17 ] 

18 

19 echo '< /table> "7 

20 mysql_free_result ($ result) 7 

21 

22 }else{ 

23 echo '<p class= "error"> 很 抱 坎 发 生 系统 错误 如 下 :< /p> '; 

24 echo mysql_error () 

25 } 

26 


27 mysql_close ()， 
28 include('./includes/footer.html'); 


29 3> 
(2) 连接 并 查询 数据 库 。 


require once("./includes/mysql_connect.php"); 

$ sql= "SELECT name, DATE FORMAT (registration date, '%Y 年 $m 月 $d 日 ')AS dr FROM users ORDER 
BY registration date ASC"; 

mysql_query ("SET NAMES 'gb2312'"); 

$result=@mysql query ($ sql); 


这 里 的 查询 将 分 为 两 列 : 用 户 名 和 他 们 的 注册 日 期 ,注册 日 期 通过 DATA_ 
FORMAT 函数 格式 化 格式 为 YYYY 年 MM 月 DD 日 ,然后 通过 SQL 别名 (dr) 提 供给 
返回 的 结果 。 

(3) 显示 查询 结果 。 


if ($result){ 


第 9 章 ”使 用 PHP 和 MySQL 


echo '<table align= "center" cellspacing= "0" cellpadding= "5"> 
<tr><td><b> 姓 名 < /b>< /td><td><b> 注 册 时 间 < /b>< /td>< /tr>'; 
while ($row=mysql fetch array($ result, MYSQL ASSOC)){ 

echo '<tr><td> ' .$row['name'] . '</td><td>' . $row['dr'] . '</td></tr>'; 
} 
echo '< /table> '; 


为 了 显示 结果 ,首先 在 HTML 中 建立 了 一 个 表 和 标题 行 。 然 后 使 用 mysql_fetch_ 
array() 函 数 遍 历 结果 ,并 打印 出 后 续 的 每 一 行 。 最 后 关闭 这 个 表 。 

应 注意 的 是 ,在 while 循环 中 ,使 用 正确 的 列 名 和 别名 来 引用 返回 的 每 个 值 ，$ row 
[name]] 和 $row['dr], 不 用 引用 $row['registration_date"], 因 为 不 会 返回 这 样 的 字段 
名 称 。 

(4) 释放 查询 资源 。 


mysql_free result ($result); 


同样 ,这 一 步 是 可 选 的 ,但 是 采用 它 将 是 一 种 良好 的 编程 风格 。 
(5) 完成 条 件 语句 。 
} else { 

echo '<p class= "error"> 很 抱 坎 发 生 系统 错误 如 下 :</p> '; 

echo mysql_error () 7 


} 


当 $result 返回 为 FALSE, 则 意味 着 SQL 语句 执行 有 误 ,打印 出 MySQL 的 错误 有 
利于 调试 。 

(6) 关闭 数据 库 连 接 , 完 成 页 面 。 

mysql_close(); 

include ('./includes/footer.html'); 


?> 


(7) 将 文件 另存 为 view_users. php, 上 传 到 Web 服务 器 ,并 在 浏览 器 中 测试 它 (参见 
图 9-9) 。 

记 住 ,必须 使 用 mysql_query() 执 行 SQL 查询 ,然后 使 用 mysql_fetch_array() 来 检 
索 单行 信息 ,如 果 要 检索 多 行 , 则 可 使 用 while 循环 ,而 不 是 for 或 者 foreach 。 

需要 注意 的 是 ,如 果 需 要 在 while 循环 内 运行 第 二 个 查询 , 则 一 定 要 为 其 使 用 不 用 的 
变量 名 称 , 例 如 ,使 用 $result2 和 $row2, 如 果 再 使 用 $result 和 $row, 则 会 遇 到 逻辑 
错误 。 

另外 ,PHP 还 有 两 个 处 理 查询 结果 数组 的 函数 .分别 是 mysql_fetch_row() 函 数 和 
mysql_fetch _assoc() 困 数 。 其 中 ,mysql _fetch _row() 郴 数 与 mysql _fetch _array 
($result，MYSQL_NUM) 函数 等 价 ,mysql_fetch_assoc() 国 数 与 mysql_fetch_array 
($result， MYSQL_ASSOC) 函 数 等 价 。 
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AAA 
(Opethobe Pp - cx]| Bsmmsir | 全 六 南 


注册 时 间 

2013 年 06 月 07 日 
2013 年 08 月 07 日 
2013 年 10 月 07 日 
2014 年 02 月 07 日 
2014 年 05 月 07 日 
2014 年 05 月 12 日 
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图 9-9 从 数据 库 中 检索 所 有 的 用 户 记录 


9.3.3 统计 返回 的 记录 数 

1.，mysqlL_num_rows() 函数 

mysqlL_num_rows() 函 数 返回 SELECT 查询 检索 的 行 数 ,并 取 查 询 结果 作为 一 个 参 
数 。 用 法 如 下 : 

$ num=mysql_num rows ($result); 


mysql_num_rows() 将 返回 结果 集中 行 的 数目 。 此 命令 仅 对 SELECT 语句 有 效 。 

2. 统计 返回 的 记录 

为 了 演示 mysql_num_rows() 函数 的 用 法 ,修改 view_users. php 页 面 , 列 出 注册 用 
户 的 总 数 。 

(1) 在 文本 编辑 器 中 打开 view_users. php( 参 见 脚 本 9-5) 。 


脚本 9-5 view_users. php 将 显示 注册 用 户 的 总 数 。 


< ?php #Script 9- 5 -view_users.php (基于 脚本 9- 4 更新) 
$page title= ' 查 看 当前 所 有 用 户 '; 

include ('./includes/header.html'); 

echo '<hl id= "mainhead"> 注 册 用 户 < /hl> '; 


require once("./includes/mysql_connect.php"); 

$ sql= "SELECT name, DATE FORMAT (registration date, '$Y 年 $m 月 $d BH ')AS dr FROM users 
ORDER BY registration date ASC"; 

8 mysql query("SET NAMES 'gb2312'"); 


wow wm 


9 $result-@mysql query($ sql); 
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10 $num=mysql_num rows ($result); 

11 

12 if($num>0){ 

13 echo "<p> 共 有 $num 位 注册 用 户 < /p> \n"; 


14 echo '<table align= "center" cellspacing="0" cellpadding="5"> 

15 <tr><td><b> 姓 名 </b></td><td><b> 注 册 时 间 < /b></td></tr>'; 
16 

17 while($ row=mysql_fetch array($ result, MYSQL RSSOC)){ 

18 echo '<tr><td> ' .$row['name'] . "</td><td> ' . Srow["dr'] . "</td></tr> 7 
19 } 

20 

级 echo "< /table> '7 

22 mysq]_free_result ($ result)7 

2 

24 J}elsef{ 

25 echo '<p class= "error"> 当前 还 没有 注册 用 户 。< /p>'; 

26 } 

27 


28 mysql close(); 
29 include('./includes/footer.html'); 
30 ?> 


(2) 在 mysql_query() 函数 调用 之 后 添加 如 下 代码 : 
$num=mysql_ num rows($ result); 


这 一 行 代码 将 把 返回 的 行 数 赋予 $ num 变量 。 
(3) 把 原来 的 $result 条 件 语句 更 改 如 下 : 


if($num >0){ 


以 前 编写 的 条 件 是 基于 查询 是 否 会 工作 ,而 不 是 基于 是 否 会 返回 任何 记录 ,现在 更 


准确 。 


(4) 打印 出 注册 用 户 的 总 数 。 

echo "<p> 共 有 5 num 位 注册 用 户 < /p>\n"; 

(5) 更 改 该 条 件 语句 的 else 子 句 。 

echo '<p class="error"> 当前 还 没有 注册 用 户 。< /p> '; 


原来 的 条 件 语句 基于 查询 是 否 工作 ,应 该 已 经 通过 调试 ,可 以 不 需要 原来 的 错误 信 
现在 的 出 错 信息 只 是 指示 是 否 不 会 返回 任何 记录 。 
(6) 将 文件 另存 为 view_users. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 


(参见 图 9-10) 。 


3. 设计 字段 值 唯一 
mysql_num_rows() 函 数 的 另 一 个 使 用 方式 ,可 以 用 来 检查 数据 的 唯一 性 。 例 如 ,本 
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De 父 htpyWiocahosyboc DP - Cx] 避 asm5i# 疡 。 x[ 全 安 丽 


注册 用 户 


共有 6 位 注册 用 户 


注册 时 间 
2013 年 06 月 07 日 
2013 年 08 月 07 日 
2013 年 10 月 07 日 
George 2014 年 02 月 07 日 
Ringo ”2014 年 05 月 07 日 
李 明 2014 年 05 月 12 日 
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图 9-10 注册 用 户 的 数量 显示 页 面 顶部 


例 中 用 户 的 电子 邮件 地 址 是 唯一 的 ,不 允许 重复 的 。 下 面 的 脚本 将 测试 在 用 户 使 用 一 个 
电子 邮件 地 址 注册 时 ,该 地 址 是 否 已 经 被 使 用 。 
(1) 在 文本 编辑 器 中 打开 register. php( 参 见 脚本 9-6) 。 


脚本 9-6 register. php 脚本 现在 将 检查 电子 邮件 地 址 是 否 重复 。 


< ?php #Script 9- 6 - register.php (基于 脚本 9- 3 更 新 ) 
$page title= ' 注 册 '; 
include ('./includes/header.html'); 


入 

并 

4 

5 if(isset($_POST['Submit'])){ 
6 $errors=array(); 

7 if(empty($_POST["name"])){ 

8 S$errors[]= ' 您 忘记 输入 用 户 名 .7 
9 


} 


10 if (empty($_POST["'email"])){ 

11 $errors[]= ' 您 忘记 输入 您 的 EMAIL 地址 ."'; 

入 } 

13 if(!empty($_POST['password1'])){ 

14 if($_POST['password1'] !=$_POST['password2']){ 
15 $errors[]=' 两 次 输入 密码 不 同 ."'; 

16 } 

a }elsef{f 

18 $errors[]= ' 您 忘记 输入 密码 ."; 

19 } 
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if (empty ($errors)){ 
require once("./includes/mysql connect .php"); 
$ query= "SELECT user_id FROM users email="'{$_POST['email']}'"; 
$result=mysql_query ($ query); 
if (mysql._ num rows ($result)==0){ 


$sql="INSERT INIO 'users' ('user_id', 'name', 'password', 'email', 
'registration date') 
VALUES(NULL , '{$ _ POST['name ']}', SHAl (' {$ _ POST 
['password1"]}"'), '{$_POST['email']}', NOW());"; 
mysql_ query ("SET NAMES 'gb2312'"); 
$result=@mysql query($ sql); 


if($ result){ 
//$body=" 谢 谢 您 注册 我 们 的 站 点 !'\n 您 的 登录 密码 为 : {$ _POST 
['password1']}"; 
//mail($ _POST['email'],，' 谢 谢 您 的 注册 !'，$ body,'From: shiying@ 
zdxy.cn'); 
echo '<hl id= "mainhead"> 注 册 成 功 !< /hl>'; 
echo "<p> 谢 谢 您 注册 我 们 的 站 点 !\n 您 的 登录 密码 为 :{$ _POST 
["passwordl1']}</p> ";» 
echo '<p> 您 已 注册 成 功 , 确 认 邮 件 已 发 送 至 您 的 邮箱 !< /p><p>< /p> "'; 
include ('./includes/footer.html'); 
exit (); 
Jelse{ 
echo '<hl id= "mainhead"> 系 统 错误 !< /hl> 
<p class= "error"> 很 抱 款 您 暂时 无 法 注册 。<br />'; 
echo '<P> ' . mysql error(); 
include ('./includes/footer.html')7 
exit () 
} 
} else { 
echo '<hl id= "mainhead"> 错 误 !< /hl> 
<P class= "error"> 对 不 起 ,这 个 email 地 址 已 被 注册 ,请 重新 选择 。< /p> '; 
} 
mysql_close(); 
} else { 
echo '<hl id- "mainhead"> 错 误 !< /hl> 
<p class= "error"> 出 现 以 下 错误 :<br />'; 
foreach ($ errors as Smsg){ 
echo " - $msg<br /> \n"7 
} 
echo '< /p><p> 请 重 填 :< /p><p><br /></p>'; 
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60 } 
61 ?> 


62 <h2> 注 册 :< /h2> 
63 < form action= "register.php" method- "post"> 


64 


<table width= "299" height= "151"> 

<tr> 
<td> 用 户 名 : < /td> 
<td><input type= "text" name= "name" size="20" maxlength= "40" value="< ?php 
if(isset($_POsST['name']))echo $_POST['name']; ?>" /></td> 

</tr> 

<tr> 
<td>Email 地 址 :< /td> 
<td> < input type= "text" name= "email" size="20" maxlength="40" value="<?php if 
(isset($_POST['email']))echo $_POsT['email']; ?>" /></td> 

</tr> 

<tr> 
<td> 密 码 :< /td> 
<td>< input type= "password" name= "passwordl"” size="10" maxlength="20" /></ 
td> 

</tr> 

<tr> 
<td> 确 认 密码 : < /td> 
<td>< input type= "password" name= "password2" size= "10" maxlength="20" /></ 
td> 

</tr> 

<tr> 
<td colspan= "2"><div align= "center"> 

< input type= "submit" name="Submit" value= "注册" /> 

</div></td> 

</tr> 

</table> 


87 </fom> 


88 <?php 
89 include('./includes/footer.html'); 


30 -2% 


(2) 在 INSERT 查询 (第 26 行 ) 之 前 ,添加 如 下 代码 : 


$ query= "SELECT user id FROM users WHERE email='{$_POST['email']}'"; 


$result=mysql_query ($ query); 

if (mysql_num rows ($result)==0){ 

这 个 查询 将 通过 尝试 选择 记录 ,来 检查 提交 的 电子 邮件 地 址 目前 是 否 在 数据 库 中 ,如 
果 查 询 结 果 返 回 的 行 数 等 于 0, 则 可 以 安全 地 注册 新 用 户 。 
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(3) 在 if( $result) 条 件 语句 结束 (原始 脚本 的 第 43 行 ) 后 ,添加 如 下 语句 : 


} else { 
echo '<hl id- "mainhead"> 错 误 !< /hl> 
<p class= "error"> 对 不 起 ,这 个 email 地 址 已 被 注册 ,请 重新 选择 。< /p> '; 
} 
这 个 else 子 句 是 if(mysql_num_rows( $result) 二 二 0) 条 件 语句 的 结尾 部 分 , 它 会 
报告 已 经 获取 电子 邮件 地 址 的 记录 ,不 可 再 使 用 相同 的 地 址 注册 。 
(4) 将 文件 另存 为 register. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 ( 参 
见 图 9-11) 。 


国人 rapyiochosybocP - cx]| 于 x 枯 分 交 瘟 


错误 ! 


对 不 起 ， 这 个 emaie 址 已 被 注册 ， 请 重新 选择 。 


se 


Pary@126. com 


图 9-11 注册 过 程 不 允许 使 用 现 有 的 Email 地 址 注册 


每 个 用 户 管理 系统 都 包含 注册 及 登录 功能 ,应 该 保证 有 一 个 唯一 的 字段 ,可 以 是 电子 
邮件 地 址 或 用 户 名 。 在 登录 的 过 程 中 ,可 以 用 这 个 唯一 的 值 与 密码 一 起 结合 起 来 验证 
用 户 。 

本 脚本 通过 PHP 来 检查 字段 是 否 唯 一 ,我 们 应 该 通过 在 MySQL 中 的 字段 上 建立 
UNIQUE 唯一 索引 ,来 保证 字段 的 唯一 性 。 在 上 
phpMyAdmin 中 ,可 以 修改 表 结 构 来 完成 这 一 操作 ， 时 村 各 昌 是 关 a 有 | 关 交 | 下 的 作 本 导 主 全 

PRIMARY PRIMARY 6 Pp X userid 
在 对 应 字段 单 击 画 按钮 , 即 可 把 为 字段 添加 唯一 属 ”EGGUE 站 | 久 |Ix| ea 
性 。 一 旦 执行 了 这 个 操作 , users 表 将 会 产生 两 个 索 ”图 9-12 users 表 的 两 个 索引 项 
引 项 : 一 个 是 主键 user_id, 另 一 个 是 唯一 的 email, 如 
9-12 所 示 ,可 以 尝试 插入 会 引发 MySQL 错误 的 重复 值 来 测试 。 
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9.4 项 目 训 练 


用 户 管理 之 密码 管理 


9.4.1 项 目 说 明 


修改 密码 是 一 个 很 常见 的 功能 ,其 中 涉及 用 户 身份 验证 及 更 新 数据 库 记 录 的 相关 功 
能 。 这 里 需要 按照 用 户 的 E-mail 地 址 验证 身份 ,在 E-mail 地 址 对 应 的 旧 密 码 正确 的 前 提 
下 ,才能 修改 新 密码 ,同时 也 需要 输入 两 次 密码 ,避免 用 户 输入 错误 。 


9.4.2 设计 原理 


虽然 mysql_num_rows() 可 以 返回 查询 结果 中 的 行 数 , 但 此 命令 仅 对 SELECT 语句 
有 效 。 要 取得 被 INSERT UPDATE 或 者 DELETE 查询 所 影响 到 的 行 的 数目 ,就 要 用 
mysql_affected_rows() 函 数 。 

mysql_affected_rows() 函 数 会 返回 前 一 次 INSERT DELETE 及 UPDATE 查询 的 
记录 行 数 。 其 用 法 如 下 : 


$num=mysql affected rows ($dbc); 


这 个 函数 所 带 的 参数 是 数据 库 连接 变量 $ dbc, 这 个 参数 是 可 选 的 。 如 果 没 有 指定 这 
个 参数 ,默认 使 用 最 后 被 mysql_connect() 打 开 的 连接 。 如 果 没 有 找到 该 连接 ,函数 会 尝 
试 调用 mysql_connect() 建 立 连接 并 使 用 它 。 

mysql_affected_rows() 郴 数 取 得 最 近 一 次 与 $dbc 关联 的 INSERT、DELETE 或 
UPDATE 查询 所 影响 的 记录 行 数 。 

若 执行 成 功 , 则 返回 受 影响 的 行 的 数目 。 如 果 最 近 一 次 查询 失败 , 则 函数 返回 一 1。 

当 使 用 UPDATE 查询 时 , MySQL 不 会 将 原 值 与 新 值 一 样 的 列 更 新 。 这 样 使 得 
mysql_affected_rows() 函数 的 返回 值 不 一 定 就 是 查询 条 件 所 符合 的 记录 数 , 只 有 真正 被 
修改 的 记录 数 才 会 被 返回 。 

特别 注意 ,如 果 最 近 一 次 操作 是 没有 任何 WHERE 条 件 的 DELETE 查询 ,在 表 中 所 
有 的 记录 都 会 被 删除 ,而 mysql_affected_rows() 函 数 会 返回 0。 


9.4.3 设计 过 程 


下 面 演示 的 脚本 将 允许 用 户 修改 密码 。 它 将 体现 两 个 重要 的 思想 : 

。 针对 注册 值 检 查 提 交 的 电子 邮件 地 址 和 密码 ,这 是 登录 系统 的 关键 条 件 ; 
。 通过 把 主键 作为 参考 点 来 更 新 数据 库 记录 。 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 脚本 (参见 脚本 9-7) 。 


脚本 9-7 本 脚本 运行 一 个 UPDATE 查询 ,使 用 mysql_affected_rows() 函数 来 确认 。 


1 <?php #Script 9-7-password.php 


可 


3 
4 
与 
6 
7 
8 
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$page title= "修改 您 的 密码 '; 
include ('./includes/header.html'); 
if(isset($_POST['Submit'])){ 


$errors=array (); //Initialize error array. 
if(empty($_POsT["'email'])){ 
$errors[]= ' 您 忘记 输入 您 的 FMAIL 地 址 .7 
} 
if (empty($_POST['password'])){ 
$errors[]= ' 您 忘记 输入 当前 的 密码 ."'; 
§ 
if(!empty($_POST['password1'])){ 
if($_POST['password1'] !=$_POST['password2°']){ 
$errors[]=' 两 次 输入 密码 不 同 .'; 
} 
} else { 


$errors[]= ' 您 忘记 输入 新 密码 ."; 


if (empty ($errors)){ 
require_once("./includes/mysql._connect .php"); 
$ query= " SELECT user_id FROM users (email= '{$ _POST['email']}' AND 
password= SHA('{$_POST['password’]}"))"; 
$result=mysql_query ($ query) 7 
$num=mysql_num rows($ result) 7 


if (mysql_num rows ($result)==1){ 
$row=mysql_fetch array($ result, MYSQL_NUM); 
$ query= "UPDATE users SET password= SHA('{$ _POST['passwordl"']}')WHERE 
user_id=$ row[0]"; 
$result=@mysql_query ($ query); 


if(mysql_affected rows()==1){ 
echo '<hl id= "mainhead"> 谢 谢 您 !< /hl> 
<p> 您 的 密码 已 被 成 功 修改 。< /p><p><br /></p> 和 
include ('./includes/footer.html'); 
exit (); 
}else{ 
echo '<hl id= "mainhead"> 系 统 错误 !< /hl> 
<p class= "error"> 很 抱 款 ,修改 密码 没 用 成 功 。 发 生 如 下 错误 :< /p> '; 
echo '<p> ' . mysql error(). '</p>'; 
include ('./includes/footer.html'); 


exit (); 
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44 } else { 

45 echo '<hl id= "mainhead"> 错 误 !< /hl> 

46 <P class="error"> 对 不 起 ,您 的 电子 邮件 地 址 与 原来 的 密码 不 匹配 。< /p> '; 
47 } 

48 mysql_close(); 

49 } else { 

50 

二 echo '<hl id= "mainhead"> 错 误 :< /hl> 

52 <p class= "error"> 出 现 以 下 错误 :<br />'; 

53 foreach ($ errors as $msg){ 

54 echo " -Smsg<br /> \n"7 

55 } 

56 echo '< /p><p> 请 重 填 :< /p><p><br /></p>'; 
5 } 

58 } 

59 ?> 


60 <h2> 修 改 您 的 密码 < /h2> 
61 <formaction="password.php" method= "post"> 


62 <table width= "299" height= "151"> 

63 <tr> 

64 <td>Email 地 址 :< /td> 

65 <td> < input type= "text" name= "email" size="20" maxlength= "40" value= 
"< ?php if (isset ($_POST['email']))echo $_POST["email']; ?>" /></to> 

66 </tr> 

67 <tr> 

68 <td> 原 来 的 密码 :< /td> 

69 <td>< input type= "password" name= "password" size= "10" maxlength= 
"20" /></td> 

70 </tr> 

71 <tr> 

攻 <td> 新 密码 :< /td> 

73 <td>< input type= "password" name= "passwordl”3size= "10"”maxlength= 
"20" /></td> 

74 </tr> 

75 <tr> 

76 <td> 确 认 新 密码 : < /td> 

77 <td>< input type= "password" name= "password2" size= "10" maxlength= 
"20"”/>< /td> 

78 </tr> 

3 <tr> 

80 <td colspan= "2"><div align= "center"> 

81 < input type= "submit" name= "Submit" value= "修改 "” /> 

82 </div></td> 


83 </tr> 
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84 </table> 

85 </form> 

86 <?php 

87 include('./includes/footer.html'); 
88 ?> 


(2) 创建 主 条 件 语句 。 
if(isset($_POST['Submit'])){ 


这 个 页 面 将 同时 显示 和 处 理 表 单 。 
(3) 检查 提交 的 数据 。 


$errors=array (); //Initialize error array. 
if (empty($_POST['email'])){ 
$errors[]= ' 您 忘记 输入 您 的 EMAIL 地 址 .7 
} 
if (empty ($_POST['password'])) { 
$errors[]= ' 您 忘记 输入 当前 的 密码 .7 
} 
if(!empty($_POST['password1"'])){ 
if($_POST['password1'] !=$_POST['password2']){ 
$errors[]=' 两 次 输入 密码 不 同 ."; 
} 
} else { 
Serrors[]= ' 您 忘记 输入 新 密码 .'; 
} 


这 些 处 理 过 程 本 身 与 register. php 中 的 那些 处 理 过 程 类 似 。 表 单 将 有 4 个 输入 框 : 
电子 邮件 地 址 、 原 来 的 密码 .新 密码 和 确认 新 密码 。 
(4) 如 果 通 过 了 所 有 的 测试 ,就 会 检索 用 户 的 user_id 编号 。 


if (empty ($ errors)){ 
require once("./includes/mysql_connect .php"); 
$ query= "SELECT user_ id FROM users WHERE (email='{$_POST['email']}' AND password= SHA 
("'{$_POST['password']}"))"; 
$result=mysql query ($ query); 
Snum=mysql_num rows ($result); 
if (mysql_num_rows ($ result)==1){ 
Srow=mysql_fetch_array($ result，MYSQL NUM) 7 


第 一 个 查询 只 会 返回 与 提交 的 电子 邮件 地 址 和 密码 匹配 的 记录 的 user_id 字段 。 为 
了 比较 提交 的 密码 和 存储 的 密码 ,可 使 用 SHA10 〇 函数 再 次 对 其 加 密 。 如 果 用 户 被 注册 ， 
并 且 正 确 输入 了 电子 邮件 地 址 和 密码 ,由 于 电子 邮件 地 址 所 对 应 的 字段 具有 唯一 性 ,所 以 
会 刚好 选 出 唯一 一 条 记录 。 最 后 把 这 条 记录 作为 数组 赋予 $ row 变量 。 

(5) 更 新 数据 库 。 
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$ query= "UPDATE users SET password= SHR('{S_POST["password1']}")WHERE user id= $row[0]"; 
$result=@mysql query ($query); 


这 个 查询 将 使 用 新 的 提交 值 更 改 密码 ,其 中 user_id 列 等 于 从 前 一 个 查询 检索 出 的 


数字 。 


(6) 检查 查询 的 结果 。 


if (mysql_affected rows()==1){ 
echo '<hl id= "mainhead"> 谢 谢 您 !< /hl> 
<p> 您 的 密码 已 被 成 功 修改 。< /p><p><br /></p>"'; 
include('./includes/footer.html'); 
exit (); 

} else { 
echo '<hl id= "mainhead"> 系 统 错误 !< /hl> 
<p class= "error"> 很 抱 款 ,修改 密码 没 用 成 功 。 发 生 如 下 错误 :</p> '; 
echo '<P> ' . mysql error(). "< /p> "7 
include('./includes/footer.html'); 
exit (); 


} 
脚本 的 这 一 部 分 工作 与 register. php 脚本 类 似 。 在 这 里 ,如 果 mysql_affected_rows() 返 


回 数字 1, 就 更 新 了 记录 ,将 会 打印 出 成 功 信息 ,包括 页 面 脚注 ,并 完成 脚本 执行 。 如 果 没 
有 返回 这 个 数字 ,就 会 打印 数据 库 错误 。 


(7) 完成 条 件 语句 ,并 关闭 数据 库 连 接 。 

} else { 

echo '<hl id= "mainhead"> 错 误 !< /hl> 

<p class= "error"> 对 不 起 ,您 的 电子 邮件 地 址 与 原来 的 密码 不 匹配 。< /p> '; 


} 
mysql_close(); 


如 果 mysql_num_rows() 不 等 于 1, 那么 提交 的 电子 邮件 地 址 和 密码 和 数据 库 的 那些 


电子 邮件 地 址 和 密码 不 匹配 , 则 会 打印 这 个 错误 。 


(8) 打印 任何 出 错 消息 并 完成 PHP。 


} else { 


echo '<hl id= "mainhead"> 错 误 :< /hl> 
<p class= "error"> 出 现 以 下 错误 :<br />'; 
foreach (5 errors as $msg) { 
echo " - $msg<br /> \n™; 
echo '< /p><p> 请 重 填 :< /p><p><br /></p>"'; 
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2> 


(9) 显示 表单 。 


<h2> 修 改 您 的 密码 < /h2> 
< form action= "password.php" method= "post"> 
<table width= "299" height="151"> 
<ti> 
<tq> Email 地 址 :< /td> 
<td><input type= "text" name= "email" size="20" maxlength="40" value="<?php if 
(isset($_POST['email']))echo 5_POST["email"]; ?>" /></td> 
</tr> 
<tr> 
<td> 原 来 的 密码 :< /td> 
<td> < input type= "password" name= "password" size="10" maxlength= "20" />< /td> 
</tr> 
<tr> 
<td> 新 密码 :< /td> 
<td>< input type= "password" name= "passwordl1" size= "10" maxlength="20" />< /td> 


</tr> 
<tr> 


<td> 确 认 新 密码 : < /td> 
<td>< input type= "password" name= "password2" size= "10" maxlength="20" />< /td> 


</tr> 

<tr> 
<td colspan= "2"><div align= "center"> 

<input type="submit" name= "Submit" value= "修改 ”/> 

</div></td> 

</tr> 

</table> 
< /form> 


这 个 表单 也 带 有 3 个 不 同 的 密码 输入 框 : 原来 的 密码 .新 密码 和 确认 新 密码 ,还 有 一 


个 电子 邮件 地 址 输入 框 。 电 子 邮 件 地 址 输入 框 是 黏 性 的 ,密码 输入 框 不 能 是 黏 性 的 。 
(10) 包括 脚注 文件 。 


<?php 

include ('./includes/footer.html'); 

?> 

(11) 将 文件 另存 为 password. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 测试 它 
(参见 图 9-13 和 图 9-14) 。 

这 里 使 用 的 mysql_affected_rows() 条 件 语句 也 可 以 应 用 于 register. php 脚本 ,用 来 
查看 是 否 添加 了 一 条 记录 , 比 起 检查 if( $ result) ,这 个 条 件 更 严格 。 
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【< 同 EITTITTIRCEIET TI 全 攻 2 


图 9-13 ”成功 修改 密码 


车 htpy/ocalhosyboc PD- CX 


错误 ! 
对 不 起 ， 您 的 电子 邮件 地 址 与 原 末 的 密码 不 四 印 。 


修改 您 的 密码 


原来 的 密码 : 


CC 
CJ] 
CC | 
[ta 


9-14 输入 的 与 数据 库 中 的 内 容 不 匹配 则 不 会 更 新 密码 


本 章 小 结 
本 章 初 步 介绍 了 PHP 和 MySQL 的 综合 应 用 ,介绍 了 最 基础 的 连接 数据 库 、 执 行 简 
单 查询 和 检索 查询 结果 ,并 将 这 些 操作 应 用 在 一 个 完整 的 Web 系统 中 。 本 章 所 介绍 的 所 
有 技术 都 是 基础 性 的 技术 ,在 下 一 章 将 介绍 更 多 实用 的 新 知识 。 


重点 回顾 


1. 通过 PHP 连接 MySQL 数据 库 。 
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2. 通过 PHP 执行 INSERT、DELETE 和 UPDATE 等 简单 查询 。 
3. 执行 SELECT 查询 ,使 用 PHP 处 理 查询 结果 。 


本 章 实 训 
【 实 训 】 


利用 第 8 章 实 训 设计 的 “留言 板 ” 系 统 的 数据 库 , 完 成 简易 功能 ,要 求 能 够 进行 匿名 留 
言及 查看 所 有 留言 。 
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HTTP 协议 是 一 种 无 状态 技术 ,这 意味 着 每 个 单独 的 HTML 页 面 都 是 一 个 无 关 的 
实体 。 当 人 们 在 页 面 之 问 切换 时 , HTTP 没有 用 于 跟踪 用 户 或 保持 变量 的 方法 。 使 用 
PHP 可 以 克服 Web 的 无 状态 性 ,实现 无 状态 性 最 流行 的 技术 就 是 cookie 和 会 话 。 

cookie 出 现 以 前 ,在 Web 站 点 上 冲浪 是 一 次 没有 历史 记录 的 旅行 。 尽 管 浏览 器 会 跟 
踪 用 户 访问 过 的 页 面 ,允许 使 用 “后 退 ” 按 钮 返回 到 以 前 访问 过 的 页 面 ,但 是 服务 器 不 会 记 
录 谁 看 到 什么 内 容 。 如 果 服 务 器 不 能 够 跟踪 用 户 ,就 不 可 能 有 购物 车 或 自 定义 个 性 化 
Web 站 点 的 存在 。 

会 话 改进 了 cookie, 与 单独 使 用 cookie 相 比 ,允许 Web 程序 存储 和 检索 多 得 多 的 信 
息 。 这 两 种 技术 都 很 容易 和 PHP 一 起 使 用 。 在 本 章 中 ,将 基于 现 有 的 users 数据 表 , 应 
用 这 两 种 技术 实现 登录 系统 。 


10.1 使 用 cookie 


cookie 是 服务 器 在 用 户 的 机 器 上 存储 信息 的 一 种 方式 。 利 用 这 种 方式 ,站 点 可 以 在 
访问 期 间 记 住 或 跟踪 用 户 。 可 以 把 cookie 看 作 像 名 字 标 签 一 样 , 你 告诉 服务 器 你 的 名 
字 , 它 就 会 给 你 戴 上 一 个 标签 。 这 样 , 它 就 可 以 回 过 头 来 查阅 名 字 标 签 , 从 而 知道 你 是 谁 。 

有 些 人 对 cookie 心 存疑 虑 ,因为 他 们 认为 cookie 允许 服务 器 知道 了 他 们 太 多 的 事 
情 。 不 过 ,cookie 只 能 用 于 存储 提供 给 服务 器 的 信息 ,因此 ,与 大 多 数 其 他 在 线 技 术 相 比 ， 
cookie 的 安全 性 并 不 差 。 不 幸 的 是 ,许多 人 仍然 误解 了 这 种 技术 ,由 于 这 些 误解 可 能 会 破 
坏 Web 应 用 程序 的 功能 ,所 以 这 就 成 了 一 个 问题 。 


10.1.1 设置 cookie 

1. 使 用 setcookie( ) 函 数 

可 以 通过 setcookie() 函 数 发 送 cookie: 

setcookie (name, value); 

setcookie('name', 'Mary'); 

第 二 行 代码 在 把 cookie 发 送 到 浏览 器 时 , 带 有 名 字 name 和 值 Mary。 
通过 接着 使 用 setcookie() 函 数 , 可 以 继续 把 更 多 的 cookie 发 送 给 浏览 器 : 
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setcookie (ruser id', 263); 

setcookie('email', 'email@ example.com’'); 

与 在 PHP 中 使 用 任何 变量 时 一 样 ,在 给 cookie 命名 时 ,不 要 使 用 空格 或 标点 符号 ， 
但 是 ,要 特别 注意 使 用 正确 的 大 小 写字 母 。 

特别 要 注意 的 是 ,setcookie() 函 数 必须 位 于 二 html 二 标签 之 前 ,也 就 是 说 ,必须 在 任 
何其 他 信息 之 前 把 它们 从 服务 器 发 送 给 客户 。 万 一 服务 器 试图 在 Web 浏览 器 已 经 接收 
到 HTML 之 后 发 送 cookie, 甚 至 是 无 关 紧要 的 空白 ,都 会 导致 一 条 出 错 消息 ,并 且 不 会 发 
送 cookie。 氨 今 为 止 , 这 是 最 常见 的 与 cookie 相关 的 错误 ,但 是 很 容易 修复 它 。 在 创建 
cookie 时 ,headers already sent… 出 错 消息 实在 太 常见 。 要 注意 出 错 消息 说 了 些 什么 ,以 
便 找 出 并 修复 问题 。 

2. 发 送 cookie 


(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 10-1) 。 
脚本 10-1 login. php 脚本 会 在 成 功 登 录 时 创建 两 个 cookie。 


1 <?php #Script 10- 1- login.php 

2 if(isset($_POST['Submit"])){ 

3 $errors=array(); 

4 if (empty($_POST["email'])){ 

5 $errors[]= ' 您 忘记 输入 您 的 EMAIL 地址 ."; 

6 } 

7 if (empty($_POST['password'])){ 

8 $errors[]= "您 忘记 输入 密码 ."; 

9 } 

10 

11 if (empty ($errors)){ 

12 require once("./includes/mysql_connect .php"); 

13 $query=" SELECT user_id, name FROM users WHERE email="'{$_POST['email ']}' AND 
Password= SHA('{$_POST[ "password']}")"; 

14 mysql_query ("SET NAMES 'gb2312'"); 

15 $result=@mysql query ($ query); 

16 $row=mysql_ fetch array($ result, MYSQL NUM); 

E if($ row){ 

18 setcookie ('user_id', $row[0]); 

19 setcookie ('name', $row[1]); 

20 $url= "loggedin.php'; 

21 header ("Location: $url"); 

22 exit (); 

23 

24 } else { 

25 $errors[]=' 对 不 起 ,您 输入 的 电子 邮件 地 址 与 密码 有 误 。"'; 
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27 mysql_close(); 
28 } 

29 

30 J}else{ 

31 $errors=NULL; 
32 } 


33 $page title=' 登 录 '; 
34 include('./includes/header.html'); 


35 

36 if(!empty($errors)){ 

37 echo '<hl id= "mainhead"> 错 误 !< /hl> 

38 <P class= "error"> 出 现 以 下 错误 :<br />'; 

39 foreach ($ errors as $msg){ //Print each error. 
40 echo " - $msg<br /> \n"; 

41 } 

42 echo '< /p><p> 请 重 填 :< /p> '; 

43 } 

44 ?> 


45 <h2> 登 录 < /h2> 
46 <formaction="login.php" method= "post"> 


47 <table width= "296" height= "75"> 

48 <tr> 

49 <tr> 

50 <td> Email 地 址 :< /td> 

51 <td>< input type= "text" name= "email" size="20" maxlength= "40" /> 
</td> 

52 </tr> 

53 <tr> 

54 <td> 密 码 :< /td> 

55 <td>< input type= "password" name= "password" size= "20" maxlength= 
"20" /></td> 

56 </tr> 

57 <tr> 

58 <td colspan= "2"><div align="center"> 

59 < input type= "submit" name= "Submit" value= "注册 " /> 

60 </div></td> 

61 </tr> 


62 </table> 

63 </form> 

64 <?php 

65 include('./includes/footer.html'); 
66 ?> 


这 个 示例 将 建立 一 个 新 的 login. php 脚本 , 它 将 与 第 10 章 中 的 其 他 脚本 协同 工作 。 
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(2) 验证 表单 。 


if(isset ($_POST['Submit'])){ 
Serrors=array()7 
证 (empty($_POST["email"])){ 
Serrors[]= ' 您 忘记 输入 您 的 EMRAIL 地 址 .7 
} 
if (empty($_POST['password'])){ 
$errors[]= ' 您 忘记 输入 密码 .'7 
} 


这 个 脚本 将 做 两 件 事 : 处 理 表单 提交 和 显示 表单 。 这 个 条 件 语 句 检 查 提交 。 主 条 件 
语句 检查 是 否 已 提交 了 表单 ,然后 检查 电子 邮件 地 址 和 密码 输入 框 的 值 。 
(3) 连接 数据 库 , 从 数据 库 中 检索 这 个 用 户 的 user_id 和 name。 


if (empty ($errors)){ 
require once("./includes/mysql_connect .php"); 
$ query= "SELECT user_id, name FROM users WHERE email='{$_POST['email']}' AND password 
=SHA('{$_POST['password’]}"')"; 
mysql_query ("SET NAMES 'gb2312'"); 
$result=@mysql query ($query); 
$row=mysql_ fetch array($ result, MYSQL NUM); 

如 果 通 过 了 两 个 验证 测试 ,就 会 查询 数据 库 ,检索 记录 的 user_id 和 name 值 ,其 中 
email 列 与 提交 的 电子 邮件 地 址 匹配 ,密码 则 与 提交 密码 的 加 密 版 本 匹配 。 执 行 该 查询 
语句 前 ,需要 先 转换 编码 字符 集 。 

(4) 如 果 用 户 输入 了 正确 的 信息 , 则 登录 。 


if ($row){ 
setcookie('user id', $row[0]); 


setcookie('name', $row[1]); 

仅 当 先前 的 查询 返回 至 少 一 条 记录 时 , $ row 变量 才 会 具有 一 个 值 。 在 这 种 情况 下 ， 
将 会 创建 两 个 cookie。 

(5) 把 用 户 重 定向 到 另 一 个 页 面 。 

$url .= '1oggedin.php' 

header ("Location: $url"); 

exit(); 

要 重 定向 到 的 特定 页 面 是 loggedin. php。 在 header() 函 数 中 使 用 相对 URL, 并 用 
exit() 终 止 脚本 的 执行 。header() 函数 的 用 法 已 在 第 6 章 介 绍 过 。 

(6) 完成 $row 条 件 语句 和 $ errors 条 件 语 句 , 并 关闭 数据 库 连 接 。 


}else { 
Serrors[]= ' 对 不 起 ,您 输入 的 电子 邮件 地 址 与 密码 有 误 。'; 
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} 
mysql_close(); 
3 


这 个 脚本 的 错误 管理 与 register. php 脚本 中 的 错误 管理 非常 相似 。 由 于 在 调试 
setcookie() 和 header() 这 几 行 代码 之 前 ,不 能 向 浏览 器 发 送 任何 内 容 , 所 以 必须 把 错误 保 
存 起 来 ,以 后 再 打印 出 来 。 

(7) 完成 主 提交 条 件 语句 ,包含 HTML 头 部 ,打印 出 错误 信息 。 


} else { 
$errors=NULL; 
} 
$page_ title= ' 登 录 '; 
include ('./includes/header.html'); 


if(!empty($ errors)){ 
echo '<hl id= "mainhead"> 错 误 !< /hl> 
<p class= "error"> 出 现 以 下 错误 :<br />'; 
foreach ($ errors as Smsg){ //Print each error. 
echo " -$msg<br />\n"; 
} 
echo '< /p><p> 请 重 填 :< /p> '; 
} 
营区 


同样 ,这 一 步 以 及 前 面 的 步骤 都 与 register. php 脚本 的 那些 步骤 相似 。 第 一 个 else 
条 件 语句 把 $ errors 变量 设置 为 NULL, 指 示 在 第 一 次 运行 这 个 页 面 时 不 需要 打印 任何 
错误 。 然 后 会 设置 页 面 的 标题 ,并 包含 模板 的 头 文件 。 最 后 将 打印 来 自 表单 提交 的 任何 
现 有 的 错误 。 

(8) 显示 HTML 表单 。 


<h2> 登 录 < /h2> 
< form action= "login.php" method= "post"> 
<table width= "296" height="75"> 
<tr> 
<tr> 
<td> Email 地 址 :< /td> 
<td> < input type= "text" name= "email" size="20" maxlength= "40" />< /td> 
</tr> 
<tr> 
<td> 密 码 :< /td> 
<td>< input type= "password" name= "password" size= "20" maxlength= "20" /> 
</to> 
</tr> 
<try 
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<td colspan= "2"><div align= "center"> 
<input type= "submit" name= "Submit" value= "注册 " /> 
</div></td> 
</tr> 
</table> 
< /form> 


这 个 HTML 表单 使 用 表格 排版 , 带 有 两 个 输入 框 ,用 于 输入 电子 邮件 地 址 和 密码 ， 
并 把 数据 交 回 这 个 相同 的 页 面 。 

(9) 包含 PHP 脚注 。 

<?Pphp 

include ('./includes/footer.html'); 

2> 

(10) 将 文件 另存 为 login. php, 上 传 到 Web 服务 器 ,并 在 Web 浏览 器 中 加 载 这 个 页 
面 ,如 图 10-1 所 示 。 


CD[@ hpi/ocalhosyboe P - OX 


© Copyright 2014 


图 10-1 登录 表单 


注意 : cookie 被 限制 为 总 共 包 含 大 约 4KB 的 数据 ,每 个 Web 浏览 器 可 以 记 住 来 自任 
何 一 个 站 点 的 有 限 数量 的 cookie。 对 于 目前 的 大 多 数 Web 浏览 器 ,这 个 限制 是 50 个 
cookie。 

因为 不 同 的 浏览 器 将 以 不 同 的 方式 处 理 cookie,PHP 中 有 几 个 函数 可 以 在 不 同 的 浏 
览 器 中 生成 不 同 的 结果 ,setcookie() 函数 是 其 中 之 一 。 一 定 要 在 不 同 平 台 上 的 多 个 浏览 
器 中 测试 Web 站 点 ,以 确保 一 致 性 。 


10.1.2 访问 cookie 


要 从 cookie 中 检索 一 个 值 .只 需要 把 合适 的 cookie 名 称 用 作 键 来 引用 $_COOKIE 
超 全 局 数组 ,就 像 利用 任何 数组 一 样 。 例 如 ,为 了 从 用 如 下 一 行 代码 建立 的 cookie 中 检 
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索 一 个 值 : 
setcookie ("username', 'Mary'); 


使 用 $_COOKIE[L'username"] 将 引用 该 cookie 的 值 。 

在 下 面 的 示例 中 ,将 以 两 种 方式 来 访问 login. php 脚本 设置 的 cookie。 首 先 ,将 检查 
用 户 是 否 已 登录 ,如 果 未 登录 ,就 不 能 访问 到 这 个 页 面 ;如果 已 登录 , 则 通过 用 户 的 名 字 来 
问候 他 们 ,他 们 的 名 字 存 储 在 一 个 cookie 中 。 

(1) 在 文本 编辑 器 中 创建 一 个 新 的 PHP 文档 (参见 脚本 10-2) 。 


脚本 10-2 loggedin. php 脚本 基于 一 个 存储 的 cookie 向 用 户 打印 一 个 问候 。 


1 <?php #Script 10- 2- loggedin.php 
2 if(!isset($_COOKIE['user id'])){ 
3 $url= 'index.php'7 
4 header ("Location: $url"); 
5 exit(); 
6 上】 

7 

8 $page title=' 登 录 成 功 '; 

9 include('./includes/header.html'); 


11 ”echo "<hl> 登 录 成 功 < /hl> 
12 <p> 感谢 您 的 登录 ,，{$_COOKIE['name']}!< /p> 
13 <p><br /><br /></p>"; 


14 
15 include('./includes/footer.html'); 
26 23 


在 用 户 成 功 登录 后 ,将 把 他 们 重 定向 到 这 个 页 面 。 它 将 打印 一 个 特定 于 某 个 用 户 的 
问候 。 

(2) 检查 cookie 是 否 存在 。 

if(!isset($_COOKIE['user id'])){ 


因为 在 用 户 登 录 前 不 希望 用 户 能 访问 这 个 页 面 ,所 以 应 该 首先 检查 是 否 已 经 设置 了 
cookie。 在 login. php 中 , 若 登录 成 功 , 则 设置 $_COOKIE[user_idq] 的 值 。 
(3) 如 果 用 户 未 登录 ,就 重 定 向 他 们 。 


Surl= 'index.php'; 
header ("Location: $url"); 
exit (); 


} 


如 果 用 户 没 有 登录 ,就 会 自动 把 他 们 重 定向 到 这 个 主页 面 。 这 是 一 种 限制 访问 内 容 
的 简单 方式 。 
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(4) 包含 页 面 的 标题 。 
$page title= ' 登 录 成 功 '; 
include ('./includes/header.html'); 


(5) 使 用 cookie 欢迎 用 户 。 


echo "<h1> 登 录 成 功 < /hl> 

<pP> 感 谢 您 的 登录 ,，{$5_COOKIE['name']}!< /p> 

<p><br /><br /></p> "7 

为 了 用 名 字 问 候 用 户 , 引 用 了 $ _COOKIE['mname"] 变 量 , 括 在 花 括号 内 以 避免 解析 
错误 。 

(6) 完成 HTML 页 面 。 

include ('./includes/footer.html'); 

?> 

(7) 将 文件 另存 为 loggedin. php ,存放 在 Web 目录 中 ,在 与 login. php 相同 的 文件 来 
中 ,并 通过 login. php 登录 在 Web 浏览 器 中 测试 它 , 如 图 10-2 所 示 。 


DIG httpi//localhos/bo PD» OX 


感谢 您 的 登录 ， 李 明 ! 


图 10-2 如 果 使 用 正确 的 电子 邮件 地 址 和 密码 , 则 在 登录 后 ,会 重 定向 到 这 里 


因为 这 些 示 例 使 用 与 第 9 章 中 相同 的 数据 库 , 应 该 能 够 使 用 当时 提交 的 注册 用 户 名 
和 密码 来 登录 。 

直到 重新 加 载 了 设置 页 面 (例如 ,login. php) 或 者 访问 了 另 一 个 页 面 ( 换 句 话说 ,不 能 
设置 和 访问 同一 个 页 面 中 的 cookie) , 才 可 以 访问 cookie。 

如 果 用 户 拒 绝 一 个 cookie, 或 者 把 他 们 的 Web 浏览 器 设置 成 不 接受 它们 , 则 会 把 用 
户 自动 重 定向 到 这 个 示例 中 的 主页 ,即使 他 们 成 功 登 录 也 是 如 此 。 由 于 这 个 原因 ,可 能 想 
让 用 户 知 道 何 时 需要 cookie。 


10.1.3 删除 cookie 


虽然 在 关闭 用 户 的 浏览 器 时 cookie 会 自动 到 期 ,但 是 ,有 时 希望 手动 删除 cookie。 
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例如 ,在 具有 登录 能 力 的 Web 站 点 内 ,可 能 希望 在 用 户 注 销 时 删除 任何 cookie。 

尽管 setcookie() 函数 可 以 带 最 多 7 个 参数 ,但 是 实际 上 只 有 一 个 参数 (cookie 名 称 ) 
是 必需 的 。 如 果 发 送 一 个 包含 名 称 但 没有 值 的 cookie, 其 效果 就 相当 于 删除 现 有 的 同名 
cookie。 例 如 ,要 创建 name cookie, 可 以 使 用 下 面 这 一 行 代 码 : 

setcookie('name', 'Mary'); 

要 删除 name cookie, 可 以 编写 如 下 代码 : 

setcookie('name', ''); 

在 PHP 中 使 用 “setcookie ($cookiename，'');” 或 者 “setcookie ( $ cookiename， 
NULL); ”都 会 实现 PHP 删除 cookie。 

为 了 演示 这 些 操作 ,将 给 站 点 添加 注销 能 力 。 指 向 注销 页 面 的 链接 将 出 现在 
loggedin. php 上 。 作 为 一 种 附加 特性 ,将 更 改 头 文件 ,使 得 当 用 户 登 录 时 显示 “注销 " 链 
接 , 并 且 当 用 户 注销 时 显示 “登录 ”链接 。 

1. 删除 cookie 

(1) 在 文本 编辑 器 或 IDE 中 创建 一 个 新 的 PHP 文档 (参见 脚本 10-3)。 


脚本 10-3 logout. php 脚本 会 删除 以 前 建立 的 cookie。 


1 <?php #Script 10- 3- logout.php 
2 if(!isset($_ COOKIE['user id'])){ 
Ei 

4 $url= 'index.php'7 

5 header ("Location: $url"); 
6 exit();» 

7 

8 } else { 

9 setcookie('name', '');} 

10 setcookie ('user_id', ''); 
11 } 


13 $page title= ' 注 销 '; 
14 include('./includes/header.html'); 


16 echo "<hl> 注 销 < /hl> 
17 <p> 您 已 注销 成 功 。{$_CoOKIE['name']}!< /p> 
18 <p><br /><br /></p> "7 


20 include('./includes/footer.html'); 
2 2% 
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(2) 检查 user_id cookie 是 否 存在 ;如 果 它 不 存在 ,就 重 定向 用 户 。 


if(!isset($ COOKIE['user id'])){ 
$url= 'index.php'; 
header ("Location: $url"); 


exit (); 
与 loggedin. php 页 面 一 样 ,如 果 用 户 还 没有 登录 ,这 个 页 面 就 应 该 把 用 户 重 定向 到 
主页 。 尝 试 注销 尚未 登录 的 用 户 没 有 意义 ,因为 无 法 注销 尚未 登录 的 用 户 。 
(3) 如 果 cookie 存在 , 则 删除 它们 。 


} else { 
Setcookie('name',''); 
Setcookie('user id',''); 

} 

如 果 用 户 已 经 登录 ,这 两 个 cookie 将 有 效 地 删除 现 有 的 cookie。 

(4) 建立 PHP 页 面 的 提醒 。 

$page title= ' 注 销 '; 

include ('./includes/header.html')7 

echo "<hl> 注 销 < /hl> 

<p> 您 已 注销 成 功 。{$_CoOKIE['name']}!< /p> 

<p><br /><br />< /p>"; 

include ('./includes/footer.html'); 

2> 


这 个 页 面 本 身 也 与 loggedin. php 页 面 非常 相似 。 尽 管 它 看 起 来 似乎 有 些 奇怪 ,但 是 
因为 仍然 可 以 引用 name cookie( 刚 才 在 这 个 脚本 中 删除 了 它 ) ,考虑 以 下 过 程 很 有 意义 : 

Q 这 个 页 面 将 被 客户 请 求 。 

@ 服务 器 从 客户 的 浏览 器 读 取 合适 的 cookie。 

@ 运行 页 面 ,并 做 它 的 事情 (包括 发 送 新 的 cookie, 删 除 现 有 的 cookie) 。 

因此 , 简 言 之 ,在 第 一 次 运行 这 个 脚本 时 , 它 就 可 以 使 用 原来 的 name cookie 数据 。 
这 个 页 面 发 送 的 cookie 集 ( 删 除 的 cookie) 不 能 为 这 个 页 面 所 用 ,因此 原来 的 值 仍 然 
有 用 。 

(5) 将 文件 另存 为 logout. php ,并 存放 在 Web 目录 中 (在 与 login. php 相同 的 文件 
夹 中 ) 。 

2. 创建 注销 链接 

(1) 在 文本 编辑 器 或 IDE 中 打开 header. html( 参 见 脚本 10-4) 。 


脚本 10-4 根据 用 户 的 当前 状态 ,header. html 文件 现在 将 显示 登录 或 注销 链接 。 


1 < !DOCTYPE html PUBLIC "~ //W3C/V/DTD XHTML 1.0 Transitional//EN" 
2 "http://www.w3.0rg/TR/zxhtml1/DTD/xhtmll- transitional.dtd"> 
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(2) 将 链接 做 了 修改 ,不 是 在 导航 区 域 中 建立 一 个 永久 的 登录 链接 ,而 是 在 用 户 登 录 
时 ,让 其 显示 “注册 ”和 “登录 ”的 链接 ,或 者 如 果 用 户 没 有 登录 , 则 显示 “查看 用 户 ”“ 修 改 


<html xmlns= "http://www.w3.0rg/1999/xhtml"> 
<head> 
<meta http- equiv= "content- type" content="text/html; charset=gb2312" /> 
<title><?php echo $page title;?></title> 
<link href="includes/layout .css" rel="stylesheet" type= "text/css" /> 
< /head> 


<body> 
<div id= "wrapper"> 
<div id= "content"> 
<div id= "nav"> 
<h3> 请 选择 < /h3> 
<ul> 
<1i class="navtop"><a href="index.php" title=" 首 页 "> 首页 </a></1i> 
<?php 
if((isset ($_COOKIE['user_ id']))&s (!strpos ($_SERVER['PHP_SELF'], 'logout .php'))){ 
echo '<li><a href= "view users.php" title=" 查 看 用 户 "> 查看 用 户 </a></li>'; 
echo '<1i><a href= "password.php" title= "修改 密码 "> 修改 密码 < /a>< /li>'; 
echo'<1i><a href= "logout .php" title= "注销 "> 注销 </a></li>'; 
} else { 
echo '<1i><a href= "register.php" title= "注册 "> 注册 </a></li>'; 
echo '<1i><a href="login.php" title= "登录 "> 登录 < /a></li>'; 


?> 
</ul> 
</div> 
<!--Script 10- 4 -header.html - -> 


密码 ”和 * 注 销 ” 的 链接 。 根 据 cookie 存在 与 否 来 完成 上 面 的 条 件 语句 。 


由 于 logout. php 脚本 一 般 会 显示 一 个 注销 链接 ,因为 第 一 次 查看 注销 页 面 时 cookie 


存在 ,条 件 语句 不 得 不 检查 当前 页 面 是 否 是 logout. php 脚本 。 利 用 strpos() 函数 可 以 轻 
松 完成 这 个 任务 ,用 于 在 一 个 字符 串 内 发 现 另 一 个 字符 串 。 

(3) 保存 文件 ,将 其 存放 在 Web 目录 中 (放置 在 includes 文件 夹 内 ) ,并 在 Web 浏览 
器 中 测试 登录 /注销 过 程 ,如 图 10-3 和 图 10-4 所 示 。 


在 删除 一 个 cookie 时 ,应 该 总 是 使 用 与 设置 cookie 相同 的 参数 。 如 果 在 创建 cookie 


时 设置 了 主机 和 路 径 , 则 在 删除 cookie 中 要 再 次 使 用 它们 。 


记 住 ,直到 重新 加 载 页 面 或 者 访问 了 另 一 个 页 面 时 ,cookie 的 删除 才 会 生效 。 换 句 话 


说 ,在 那个 页 面 删除 了 cookie 之 后 , 它 仍然 可 供 页 面 使 用 。 
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me 
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图 10-3 在 用 户 登录 后 ,页 面 的 链接 


el) 


OB -ox EE x 而 全 立意 


注销 


您 已 注销 成 功 。 李 明 ! 


图 10-4 注销 后 的 结果 


10.2 使 用 session 会 话 


当 访 问 设计 良好 的 网 站 时 ,会 感觉 到 极 佳 的 连贯 性 ,就 像 翻 看 图 书 或 杂志 里 的 页 面 一 
样 。 所 有 的 一 切 都 组 合成 一 个 连贯 的 实体 。 现 实情 况 却 大 相 径 庭 。 各 个 页 面 的 每 一 部 分 
都 是 由 Web 服务 器 单独 存储 和 处 理 的 。 除 了 需要 知道 把 相关 文件 发 送 到 哪里 之 外 ,服务 
器 对 于 你 是 谁 并 不 感 兴趣 ,一 旦 脚本 执行 完成 ,通常 就 会 丢弃 PHP 变量 。POST 和 GET 
数组 中 的 变量 甚至 只 会 在 从 一 个 页 面 传递 给 另 一 个 页 面 这 段 时 间 里 存在 。 为 了 把 信息 传 
递 给 更 多 的 页 面 ,不 得 不 把 它 存储 在 表单 隐藏 字段 中 ,并 且 仅 当 表单 提交 时 它 才 会 存在 。 

为 了 解决 这 些 问 题 ,PHP( 与 其 他 的 服务 器 端 语言 一 样 ) 使 用 会 话 。 会 话 通过 在 Web 
服务 器 和 访问 者 的 计算 机 上 存储 随机 标识 符 ( 作 为 cookie) 来 确保 连贯 性 。 由 于 标识 符 对 
于 每 个 访问 者 是 唯一 的 ,在 会 话 变量 中 存储 的 所 有 信息 都 与 那个 访问 者 直接 相关 ,并 且 不 
能 被 其 他 任何 人 看 见 。 

使 数据 可 供 Web 站 点 上 的 多 个 页 面 使 用 的 另 一 种 方法 是 使 用 会 话 (session) 。 会 话 
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假定 数据 存储 在 服务 器 上 ,而 不 是 在 Web 浏览 器 中 ,会 话 标 识 符 用 于 定位 特定 用 户 的 记 
录 ( 会 话 数据 )。 这 个 会 话 标识 符 通常 通过 cookie 存储 在 用 户 的 Web 浏览 器 中 ,但 是 , 敏 
感 数据 本 身 ( 如 用 户 的 ID、 姓 名 等 ) 总 是 保留 在 服务 器 上 。 

有 人 可 能 会 问 : 本 来 cookie 工作 得 好 好 的 ,为 什么 还 要 使 用 会 话 ? 首先 ,会 话 更 安 
全 ,这 是 由 于 所 有 记录 的 信息 都 存储 在 服务 器 上 .并 且 不 会 持续 不 断 地 在 服务 器 和 客户 之 
间 来 回 发 送 。 其 次 ,可 以 在 会 话 中 存储 更 多 的 数据 。 第 三 ,有 些 用 户 拒绝 cookie, 或 者 完 
全 关闭 它们 。 虽 然 会 话 被 设计 成 与 cookie 一 起 工作 ,但 它 也 可 以 独立 起 作用 。 


10.2.1 设置 session 变量 


关于 会 话 的 最 重要 的 规则 是 ,将 会 使 用 它们 的 每 个 页 面 首先 都 必须 调用 session_ 
start() 函 数 。 这 个 函数 告诉 PHP 开启 一 个 新 的 会 话 , 或 者 访问 一 个 现 有 的 会 话 。 必 须 
在 把 任何 内 容 发 送 到 Web 浏览 器 之 前 调用 这 个 函数 。 

第 一 次 使 用 session_start() 函 数 时 , 它 会 试图 发 送 一 个 cookie, 名 称 为 PHPSESSID 
(会 话 名 称 ) ,和 一 个 类 似 于 a61f8670baa8e90a30c878df89a2074b 的 一 串 值 , 是 32 个 十 六 
进 制 字母 , 它 就 是 会 话 ID。 由 于 试图 发 送 一 个 cookie, 所 以 在 把 任何 数据 发 送 到 Web 浏 
览 器 之 前 ,必须 先 调 用 session_start() ,这 与 使 用 setcookie() 和 header() 这 两 个 函数 时 的 
情况 一 样 。 

一 旦 启动 了 会 话 , 就 可 以 使 用 正常 的 数组 语法 把 值 注册 到 会 话 中 : 

$_SESSION['key']=value; 

$_SESSION['name']= "Mary'7 

$_SESSION['id']=48; 

记 住 这 一 点 后 ,让 我 们 更 新 login. php 脚本 ,实现 开启 会 话 功 能 。 

(1) 在 文本 编辑 器 或 IDE 中 打开 login. php( 参 见 脚本 10-5)。 


脚本 10-5 login. php 脚本 现在 使 用 的 是 会 话 ,而 不 是 cookie。 


1 <?php #Script 10- 5- login.php 

2 if(isset($_POST['Submit"])){ 

3 $errors=array(); 
ifE(empty($_POST["email"])){ 

5 S$errors[]= ' 您 忘记 输入 您 的 PaAIL 地 址 .7 
6 } 

可 if (empty($_POST['password'])){ 

8 $errors[]= ' 您 忘记 输入 密码 .'; 

9 } 

10 

11 if (empty ($errors)){ 

12 require once("./includes/mysql connect .php"); 
13 mysql query ("SET NAMES 'gb2312'"); 


14 $ query= "SELECT user id, name FROM users WHERE email= '{$_POST['email']}"' AND 
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Password= SHA('{$_POST['password"]}")"; 
$result=@mysql query($ query); 
$row=mysql fetch array($ result, MYSQL NUM); 
if($ row){ 
session_start (); 
$_SESSION['user_id']=$ row[0]; 
$_SESSION['name']=$ row[1]; 


$url= 'loggedin.php'; 
header ("Location: $url"); 


exit (); 


} else { 
S$errors[]= ' 对 不 起 ,您 输入 的 电子 邮件 地 址 与 密码 有 误 。'7 
} 
mysql_close(); 
} 


} else { 
$errors=NULL; 
} 
Spage _title= ' 登 录 '; 
include('./includes/header.html'); 


if(!empty ($errors)){ 
echo '<hl id= "mainhead"> 错 误 !< /hl> 
<p class= "error"> 出 现 以 下 错误 :<br />'; 
foreach ($ errors as Smsg){ //Print each error. 
echo " - $msg<br /> \n"7 
} 
echo '< /p><p> 请 重 填 :< /p> '; 


?> 
<h2> 登 录 < /h2> 
< form action= "login.php" method= "post"> 
<table width= "296" height= "75"> 
<tr> 
<tr> 
<td>Email 地 址 :< /td> 
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<td><input type= "text" name= "email" size= "20" maxlength="40" /> 


</td> 
</tr> 
<tr> 


237 


238 


PHP+MySQL 项 目 实例 开发 


57 


69 


<td> 密 码 :< /td> 
<td> < input type= "password" name= "password" size= "20" maxlength= 
"20" />< /td> 
</tr> 
<tr> 
<td colspan="2"><div align="center"> 
< input type= "submit" name= "Submit" value= "注册 " /> 
</div></td> 
</tr> 
</table> 
< /fom> 
<?php 
include('./includes/footer.html'); 
?> 


(2) 用 以 下 代码 替换 setcookie 那 几 行 代码 : 


session _ start()7 

$_SESSION['user id']=$ row[0]; 

$_SESSION['name']=$ row[1]; 

第 一 步 是 开启 会 话 。 因 为 在 脚本 中 这 一 刻 之 前 没有 echo() 语 名 .HTML 代码 或 者 
空白 ,所 以 现在 可 以 安全 地 使 用 session_start()。 然后 ,把 两 个 键 - 值 (key-value) 对 添加 
到 $_SESSION 超 全 局 数组 中 ,以 把 用 户 的 名 字 和 用 户 ID 注册 到 会 话 中 。 

(3) 将 页 面 男 存 为 login. php ,存放 在 Web 目录 中 ,并 在 Web 浏览 器 中 测试 它 , 如 
图 10-5 所 示 。 尽 管 需要 重 写 loggedin. php 及 其 头 部 和 脚本 ,但 仍然 可 以 测试 登录 脚本 。 


DT 一 
儿 爸 htpy/localhosyboc 只 - CX 


[Lininge@exanple. con 
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图 10-5 登录 表单 对 最 终 用 户 保持 不 变 ,但 是 底层 的 功能 现在 使 用 的 是 会 话 


由 于 会 话 通常 会 发 送 和 读 取 cookie, 应 该 总 是 设法 在 脚本 中 尽 可 能 早 地 开启 它们 。 
这 样 做 将 有 助 于 避免 如 下 问题 : 试图 在 已 经 发 送 头 部 (HTML 或 空白 ) 之 后 发 送 cookie。 
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如 果 愿 意 ,可 以 在 php. ini 文件 中 把 session. auto_start 设置 为 1, 从 而 不 必 在 每 个 页 
面 上 使 用 session_start() 。 这 样 做 会 加 大 服务 器 上 的 开销 ,由 于 这 个 原因 ,如 果 不 考 虑 某 
些 环境 因素 ,就 不 应 该 使 用 它 。 

可 以 在 会 话 中 存储 数组 (使 $_SESSION 成 为 一 个 多 维 数组 ) ,就 像 可 以 存储 字符 串 
或 数字 一 样 。 


10.2.2 访问 session 变量 


一 旦 启动 了 会 话 ,并 向 其 注册 了 变量 ,就 可 以 创建 将 访问 这 些 变量 的 其 他 脚本 。 为 了 
执行 该 操作 ,每 个 脚本 首先 必须 再 次 使 用 session_start() 来 启用 会 话 。 

这 个 函数 将 允许 当前 脚本 访问 以 前 启动 的 会 话 ,或 者 创建 一 个 新 的 会 话 。 要 理解 如 
果 不 能 找到 当前 会 话 ID 或 者 生成 了 新 的 会 话 ID, 那 么 在 旧 会 话 ID 下 存储 的 所 有 数据 都 
将 不 可 用 。 

假定 访问 当前 会 话 没有 问题 ,要 引用 一 个 会 话 变量 ,可 使 用 $_SESSION['var] ,就 像 
引用 任何 其 他 数组 一 样 。 

(1) 在 文本 编辑 器 或 IDE 中 打开 loggedin. php( 参 见 脚本 10-6) 。 


脚本 10-6 loggedin. php 可 以 引用 $_SESSION ,而 不 是 引用 $_COOKIE。 


< ?php #Script 10- 6- loggedin.php 
session start() 7 
if(!isset($_SESSION['user_ id'])){ 
$url= 'index.php'7 
header ("Location: $url"); 
exit (); 


} 


$page title= ' 登 录 成 功 '; 
include('./includes/header.html'); 


echo "<hl> 登 录 成 功 < /hl> 
<P> 感 谢 您 的 登录 ，{$_SESSION['name'])!< /p> 
<p><br /><br /></p> "7 


include('./includes/footer.html'); 
> 


(2) 添加 对 session_start() 函 数 的 调用 。 


session start (); 


设置 或 访问 会 话 变量 的 每 个 PHP 脚本 都 必须 使 用 session_start() 函 数 。 必 须 在 包 
含 header. html 文件 之 前 以 及 在 把 任何 内 容 发 送 到 Web 浏览 器 之 前 调用 这 一 行 代码 。 
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(3) 用 $_SESSION 替换 对 $_COOKIE 的 引用 。 
ifE(!isset($5_SESSION['user id'])){ 

和 
<p> You are now logged in, {$_SESSION['first name']}!</p> 


要 把 一 个 脚本 从 cookie 转换 成 会 话 ,只 需要 把 使 用 的 $_COOKIE 更 改 成 $_ 
SESSION( 如 果 名 称 相同 的 话 ) 。 

(4) 将 文件 另存 为 loggedin. php ,存放 在 Web 目录 中 ,并 在 浏览 器 中 测试 它 , 如 
图 10-6 所 示 。 

(5) 在 header. html 中 用 $_SESSION 替换 对 $_COOKIE 的 引用 。 


证 ((isset($_SESSION['user id']))&&(!strpos ($_SERVER["PHP SELF'], 'logout.php'))){ 


脚本 10-7 header. html 文件 现在 也 会 引用 $_SESSION ,而 不 是 引用 $_COOKIE。 


1 <!DOCTYPE html PUBLIC "~ //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.o0org/TR/xhtmll/DTD/xhtmll- transitional .dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

各 <meta http- equiv= "content— type" content="text/html; charset=gb2312" /> 

6 <title><?php echo $page title;?></title> 

7 <link href="includes/layout.css" rel="stylesheet" type="text/css" /> 
8 </head> 

9 


10 <body> 

11 <div id="wrapper"> 

12 <div id= "content"> 

13 <div id= "nav"> 

14 <h3> 请 选择 < /h3> 

15 <ul> 

16 <1i class= "navtop"><a href="index.php" title= "首页 "> 首页 </a>< /li> 
17 <?php 

18 if((isset($_SESSION['user id']))&&(!strpos($_SERVER['PHP_SELF'], 'logout .php'))){ 
19 echo '<1li><a hre 人 "view users.php" title= "查看 用 户 吃 查看 用 户 </a>< /li>'; 

20 echo '<1i><a href= "password.php" title= "修改 密码 "> 修改 密码 < /a>< /1i>'; 

> echo '<1i><a href="logout.php" title= "注销 "> 注销 </a>< /li>'; 

22 }else{ 

23 echo '<1i><a href= "register.php" title= "注册 "> 注册 < /a>< /li>'; 

24 echo '<1i><a href="login.php" title= "登录 "> 登录 < /a></li>'; 
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29 <!--Script 10- 7-header.html -一 > 

为 了 让 “登录 ”和 “注销 "等 链接 正确 地 工作 ,必须 把 头 文件 内 对 cookie 变量 的 引用 转 
变 成 会 话 。 头 文件 不 需要 调用 session_start() 函 数 ,因为 它 会 被 调用 该 聘 数 的 页 面包 括 
在 内 。 

(6) 保存 头 文件 ,存放 在 Web 目录 中 (在 includes 文件 夹 中 ) ,并 在 浏览 器 中 测试 它 ， 
如 图 10-6 所 示 。 
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10-6 登录 成 功 提示 与 链接 


注意 为 了 让 “登录 ”和 “注销 ”等 链接 在 其 他 页 面 (register. php ,index. php 等 ) 上 也 能 
正确 工作 ,将 需要 在 其 中 的 每 一 个 页 面 上 添加 session_start() 命 令 。 

如 果 你 有 一 个 应 用 程序 , 它 看 起 来 似乎 不 能 从 一 个 页 面 到 另 一 个 页 面 访问 会 话 数据 ， 
这 可 能 是 由 于 新 的 会 话 是 在 每 个 页 面 上 创建 的 。 为 了 检查 这 一 点 ,可 以 比较 会 话 ID 来 查 
看 它 是 否 相 同 。 可 以 在 发 送 会 话 cookie 时 通过 查看 会 话 cookie 来 查看 会 话 的 ID, 或 者 
使 用 session_id() 函 数 来 执行 该 操作 : 


echo session id(); 


一 旦 建立 了 会 话 变 量 ,就 可 以 使 用 它们 。 因 此 ,与 使 用 cookie 时 不 同 ,可 以 把 一 个 值 赋 予 
$ _SESSION['var"], 然 后 在 同一 个 脚本 的 后 面 引用 $_SESSION['var"]。 


10.2.3 删除 session 变量 

在 使 用 会 话 时 ,需要 创建 一 个 方法 来 删除 会 话 数据 。 当 用 户 注 销 时 ,这 是 必要 的 。 

虽然 cookie 系统 只 需要 发 送 另 一 个 cookie 来 销毁 现 有 的 cookie, 但 是 会 话 的 要 求 更 
高 ,因为 既 要 考虑 客户 上 的 cookie, 又 要 考虑 服务 器 上 的 数据 。 


要 删除 单独 一 个 会 话 变量 ,可 以 使 用 unset() 函 数 ,同时 这 个 函数 可 以 处 理 PHP 中 
的 任何 变量 : 


unset ($_SESSION['var']); 
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要 删除 每 个 会 话 变量 ,可 以 重 置 整个 $_SESSION 数组 : 
$_SESSION=array (); 

最 后 ,要 从 服务 器 中 删除 所 有 的 会 话 数据 ,可 使 用 session_destroy(): 
session destroy(); 


注意 ,在 使 用 其 中 的 任何 一 个 方法 之 前 ,页 面 都 必须 开始 于 session_start() ,这 样 做 
可 以 访问 现 有 的 会 话 。 让 我 们 更 新 logout. php 脚本 ,以 清理 会 话 数据 。 
(1) 在 文本 编辑 器 或 IDE 中 打开 logout. php( 参 见 脚本 10-8) 。 


脚本 10-8 ”销毁 会 话 需要 使 用 特殊 的 语法 。 


1 <?php #Script 10-7- logout.pPhp 
2 session start(); 

3 if(!isset($_SESSION['user id'])){ 
4 

5 $url= 'index.php'7 

6 header ("Location: $url"); 
7 exit(); 

8 

9 } else { 

10 $_SESSION=array (); 

11 session_destroy (); 

12 setcookie ('PHPSESSID', ''); 
FE 

14 


15 $page title=' 注 销 '; 
16 include('./includes/header.html'); 


18 echo "<hl> 注 销 < /hl> 
19 <p> 您 已 注销 成 功 。{$_CooKIE['name']}!< /p> 


20 <p><br /><br /></p>"; 

21 

22 include('./includes/footer.html'); 
23. 


(2) 紧 接 在 开始 PHP 行 之 后 启动 会 话 。 
session start(); 


无 论 何 时 使 用 会 话 , 都 必须 使 用 session_start() 函数 ,最 好 是 在 页 面 顶 部 这 样 做 , 即 
使 删除 会 话 也 是 如 此 。 
(3) 更 改 条 件 语句 ,使 得 它 检查 会 话 变量 是 否 存在 。 


ifE(!isset($_SESSION["user id'])){ 
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与 cookie 示例 中 的 logout. php 脚本 一 样 ,如 果 用 户 目前 没有 登录 ,就 会 重 定向 他 们 。 
(4) 用 下 面 的 几 行 代码 替换 setcookie() 行 ( 它 删除 cookie): 


$_SESSION=array (); 
session destroy(); 


setcookie('PHPSESSID', ''); 


这 里 的 第 一 行 代码 将 把 整个 $_SESSION 变量 重 置 为 一 个 新 数组 ,从 而 清除 其 现 有 的 值 。 
第 二 行 代码 从 服务 器 中 删除 数据 ,第 三 行 代码 则 会 删除 cookie。 

(5) 删除 消息 中 对 $_COOKIE 的 引用 。 

echo "<hl> 注 销 < /hl> 

<p> 您 已 注销 成 功 。< /p> 

与 使 用 logout. php 脚本 的 cookie 版 本 时 不 同 , 不 能 再 通过 用 户 的 名 字 来 引用 它们 ， 
因为 所 有 这 类 数据 都 已 被 删除 。 

(6) 将 文件 另存 为 logout. php ,存放 在 Web 目录 中 ,并 在 浏览 器 中 测试 它 , 如 图 10-7 
所 示 。 
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10-7 注销 页 面 


建议 不 要 把 $ _ SESSION 设置 成 等 于 NULL 并 且 永 远 不 要 使 用 unset($ _ 
SESSION) ,因为 它们 都 可 能 会 在 某 些 服务 器 上 引发 问题 。 

关于 会 话 包含 三 类 信息 : 会 话 标识 符 ( 它 默认 存储 在 cookie 中 ) ,会 话 数据 ( 它 存 储 在 
服务 器 上 的 一 个 文本 文件 中 ) 和 $_SESSION 数组 ( 它 指定 了 脚本 访问 文本 文件 中 的 会 话 
数据 的 方式 )。 仅 仅 删除 cookie 不 会 删除 文本 文件 ,反之 亦 然 。 清 理 $_SESSION 数组 
将 会 清除 文本 文件 中 的 数据 ,但 是 文件 本 身 将 仍然 存在 ,cookie 也 是 如 此 。 这 个 注销 脚本 
中 概括 的 三 个 步骤 实际 上 会 删除 会 话 的 所 有 痕迹 。 


10.2.4 更 改 会 话 行为 


作为 PHP 对 会 话 所 提供 支持 的 一 部 分 ,可 以 为 PHP 处 理会 话 的 方式 设置 大 约 20 种 
不 同 的 配置 选项 。 有 关 完 整 列表 ,参见 PHP 手册 ,但 是 在 这 里 重点 介绍 几 个 最 重要 的 选 
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项 。 注 意 关 于 更 改 会 话 设置 的 两 条 规则 : 

。 所 有 更 改 都 必须 在 调用 session_start() 之 前 完成 。 

。 必须 在 使 用 会 话 的 每 个 页 面 执行 相同 的 更 改 。 

可 以 在 PHP 脚本 内 使 用 ini_set() 函 数 设置 大 多 数 选项 : 

ini set (parameter, new setting); 

例如 ,如 果 需 要 使 用 会 话 cookie( 如 前 所 述 , 在 不 使 用 cookie 的 情况 下 会 话 也 可 以 工 
作 , 但 是 这 样 不 太 安全 ), 则 可 使 用 : 

ini set('session.use only cookies'，1)7 

还 可 以 更 改 会 话 的 名 称 , 这 时 可 以 使 用 session_name() 函 数 。 

session_name ("YourSession')7 


创建 自己 的 会 话 名 称 有 双重 好 处 : 它 更 安全 ,并 且 可 以 更 好 地 被 最 终 用 户 接收 ,因为 
会 话 名 称 是 最 终 用 户 将 使 用 的 cookie 名 称 。 在 删除 会 话 cookie 时 ,也 可 以 使 用 session_ 
name() 函数 ， 


setcookie (session name(), ''); 
最 后 ,还 有 一 个 session_set_cookie_params() 函 数 。 它 用 于 调整 会 话 cookie 的 设置 。 
session set_cookie params (expire, path, host, secure, httponly); 


注意 : cookie 的 到 期 时 间 只 指 cookie 在 Web 浏览 器 中 的 寿命 ,而 不 是 指 会 话 数 据 将 
在 服务 器 上 存储 多 长 时 间 。 


10.3 项 目 训 练 一 一 安全 使 用 session 


10.3.1 项 目 说 明 


本 章 前 面 介 绍 的 示例 ,它们 使 用 cookie 和 会 话 来 完成 相同 的 任务 (登录 和 注销 )。 显 
然 , 它 们 都 能 很 容易 地 在 PHP 中 使 用 ,但 是 ,真正 的 问题 是 什么 时 候 使 用 哪 一 个 更 合适 。 

与 cookie 相 比 ,会 话 具 有 以 下 优点 : 

。 一 般 更 安全 (因为 数据 保存 在 服务 器 上 ) 。 

。 允许 存储 更 多 数据 。 

。 使 用 会 话 时 .可 以 不 使 用 cookie。 

与 会 话 相 比 ,cookie 则 具有 以 下 优点 : 

。 更 容易 编程 。 

。 需要 更 少 的 服务 器 资源 。 

一 般 而 言 , 要 存储 和 检索 一 两 份 少 量 的 信息 , 则 可 使 用 cookie。 不 过 ,对 于 大 部 分 
Web 应 用 程序 ,都 会 使 用 会 话 。 
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10.3.2 项 目 原理 


由 于 重要 的 信息 通常 存储 在 会 话 中 (永远 都 不 应 该 把 敏感 数据 存储 在 cookie 中 ) ,所 
以 安全 性 变 得 更 重要 。 关 于 会 话 , 需 要 注意 以 下 两 项 : 会 话 ID 和 会 话 数 据 本 身 。 前 者 是 
一 个 指向 会 话 数据 的 引用 ,后 者 存储 在 服务 器 上 。 一 些 不 怀 好 意 的 人 极 有 可 能 通过 会 话 
ID, 而 不 是 通过 服务 器 上 的 数据 来 人 侵 一 个 会 话 , 因 此 ,在 这 里 将 重点 关注 这 方面 的 

会 话 ID 是 会 话 数 据 的 关键 。 默 认 情况 下 ,PHP 将 其 存储 在 cookie 中 ,从 安全 的 角度 
讲 ,这 是 首选 的 方法 。 在 PHP 中 可 以 在 不 使 用 cookie 的 情况 下 使 用 会 话 , 但 是 这 会 使 应 
用 程序 遭受 会 话 支持 (session hijacking) : 如 果 恶意 用 户 可 以 获悉 另 一 个 用 户 的 会 话 ID， 
就 可 以 轻松 地 欺骗 服务 器 把 它 看 作 是 他 自己 的 会 话 ID。 此 时 ,他 就 有 效 地 接管 了 原 用 户 
的 整个 会 话 , 并 且 可 以 访问 他 们 的 数据 。 因 此 ,把 会 话 ID 存储 在 cookie 中 使 得 它 更 难 被 
窃取 。 

防止 劫持 的 一 种 方法 是 : 在 会 话 中 存储 某 种 用 户 标识 符 , 然 后 反复 地 复查 这 个 值 。 
HTTP_USER_AGENT( 所 用 的 浏览 器 和 操作 系统 的 组 合 ) 是 针对 此 目的 的 一 个 可 能 的 
候选 。 这 会 增加 一 层 安全 性 ,因为 仅 当 我 运行 的 浏览 器 和 操作 系统 与 另 一 位 用 户 的 完全 
一 样 时 ,才能 够 劫持 他 的 会 话 。 


10.3.3 设计 过 程 
(1) 在 文本 编辑 器 或 IDE 中 打开 login. php( 参 见 脚本 10-9) 。 


脚本 10-9 login. php 脚本 在 会 话 中 存储 了 用 户 的 HTTP_USER_AGENT( 客 户 的 浏 


览 器 和 操作 系统 ) 的 加 密 形式 。 
1 <?php #Script 10- 8- login.php 
2 if(isset($_POST['"Submit"])){ 
3 $errors=array(); 
4 if(empty($_POST["email']))1{ 
5 $errors[]= ' 您 忘记 输入 您 的 EMAIL 地 址 .'; 
6 
过 if(empty($_POST["password'])){ 
8 $errors[]= ' 您 忘记 输入 密码 ."'; 
9 
10 
11 if (empty ($ errors)) { 
12 require once("./includes/mysql connect .php"); 
13 mysql query ("SET NAMES 'gb2312'"); 
14 $ query= "SELECT user_ id, name FROM users WHERE email='{$_POST['email']}' AND 
password= SHA('{$_POST['password"']}")"; 
15 $result=@mysql query ($ query); 


16 $ row=mysql. fetch array($ result, MYSQL NUM); 


245 


246 


PHP+MySQL 项 目 实例 开发 


51 


if($ row){ 
session_start (); 
$_SESSION['user_id']=$ row[0]; 
$_SESSION['name']=$ row[1]; 
$_SESSION['agent']=md5 ($_SERVER['HTTP_USER_RGENT']) 7 


Surl= "loggedin.php'; 
header ("Location: $url"); 


exit (); 


} else { 
$errors[]=' 对 不 起 ,您 输入 的 电子 邮件 地 址 与 密码 有 误 。'; 
} 
mysql_close(); 
} 


} else { 
$errors=NULL; 
} 
Spage _ title= ' 登 录 '; 
include ('./includes/header.html'); 


if(!empty ($errors)){ 
echo '<hl id= "mainhead"> 错 误 !< /hl> 
<p class= "error"> 出 现 以 下 错误 :<br /> 
foreach (5 errors as 5msg){ //Print each error. 
echo " - $msg<br /> \n"; 
i 
echo '< /p><p> 请 重 填 :< /p> '; 


时 实 
<h2> 登 录 < /h2> 
< form action= "login.php" method= "post"> 
<table width= "296" height="75"> 
<tr> 
<tr> 
<td> Email 地 址 :< /td> 
<td>< input type= "text" name= "email" size= "20" maxlength= "40" /> 
</td> 
</tr> 
<tr> 
<td> 密 码 :< /td> 
<td> < input type= "password" name= "password" size= "20" maxlength= 
"20" />< /td> 
</tr> 
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<tr> 
<td colspan= "2"><div align= "center"> 
< input type= "submit" name= "Submit" value= "注册 " /> 
</div></td> 
</tr> 
</table> 
</form> 
<?php 
include('./includes/footer.html')7 
3% 


(2) 在 给 其 他 会 话 变量 赋值 之 后 ,还 存储 HTTP_USER_AGENT 值 。 


$_SESSION['agent']=md5($_SERVER ['HTTP_USER RGENT'])7 


HTTP_USER_AGENT 是 $_SERVER 数组 的 一 部 分 。 它 将 具有 一 个 像 Mozilla/ 
4.0 这 样 的 值 ( 与 之 兼容 的 值 .MSIE 6.0、Windows NT 5.0、NET CLR 1.1.4322) 。 

为 了 提高 安全 性 ,这 里 没有 把 这 个 值 存储 在 会 话 中 ,而 是 用 md5() 函 数 处 理 它 。 该 
函数 基于 一 个 值 返回 32 个 字符 的 十 六 进 制 字符 串 。 从 理论 上 讲 ,任何 两 个 字符 串 都 不 具 
有 相同 的 md5() 结 果 。 

(3) 保存 文件 ,存放 在 Web 目录 中 。 

(4) 在 文本 编辑 器 中 打开 loggedin. php( 参 见 脚本 10-10) 。 


脚本 10-10 这 个 loggedin. php 脚本 现在 确认 访问 这 个 页 面 的 用 户 具 有 与 他 们 登录 时 


wm 


7 


相同 的 HTTP_ USER_ AGENT。 


< ?php #Script 10- 9- loggedin.php 
session start (); 
if(!isset($_SESSION['agent'])OR($_SESSION ['agent'] !=md5 ($_SERVER['HTTP_USER_ 
AGENT']))){ 
$url= 'index.php'7 
header ("Location: $url"); 
exit (); 
} 


$page title= ' 登 录 成 功 '; 
include ('./includes/header.html'); 


echo "<hl> 登 录 成 功 < /hl> 
<p> 感 谢 您 的 登录 ,{$_SESSION['name']}!< /p> 
<p><br /><br /></p> "7 


inclugde ('./includes/footer.html'); 
加 条 


(5) 将 ! isset($_SESSION[Luser_idJ]) 条 件 语 句 更 改 如 下 : 
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if(!isset ($_SESSION['agent'])OR($_SESSION ['agent'] !=md5($_ SERVER['HTTP USER AGENT 

"Dt 

这 个 条 件 语 句 用 于 检查 两 件 事情 。 首 先 , 它 会 查看 $_SESSION[L'agent"] 变 量 是 否 未 
设置 ,这 一 部 分 就 像 它 以 前 一 样 , 尽 管 使 用 的 是 agent 而 不 是 user_id。 这 个 条 件 语 句 的 
第 二 部 分 用 于 检查 $_SERVER['HTTP_ USER_AGENT"] 的 md5() 版 本 是 否 不 等 于 $_ 
SESSION [agent 中 存储 的 值 。 如 果 这 两 个 条 件 中 有 一 个 为 真 ,就 会 重 定向 用 户 。 

(6) 保存 这 个 文件 ,存放 在 Web 目录 中 ,并 通过 登录 在 Web 浏览 器 中 测试 它 。 

对 于 至 关 重 要 的 会 话 应 用 ,只 要 有 可 能 ,就 需要 使 用 cookie 并 通过 安全 连接 传输 它 
们 。 甚 至 可 以 通过 把 session. use_only_cookies 设置 成 1, 将 PHP 设置 成 只 使 用 cookie。 

如 果 使 用 与 其 他 域 共享 的 服务 器 ,那么 把 session. save_path 从 其 默认 设置 (可 以 被 
所 有 用 户 访问 ) 更 改 成 稍微 更 本 地 化 一 些 将 会 更 安全 。 

用 户 的 IP 地 址 不 是 一 个 良好 的 唯一 标识 符 , 这 有 两 个 原因 。 首 先 ,用 户 的 IP 地 址 可 
能 并 且 通 常会 频繁 地 发 生变 化 ,因为 ISP 在 短 时 间 内 会 动态 地 分 配 它 们 。 其 次 ,从 同一 个 
网 络 ,如 家 庭 或 办 公 室 网 络 ,访问 一 个 站 点 的 许多 用 户 可 能 都 具有 相同 的 IP 地 址 。 


本 章 小 结 


本 章 介绍 了 PHP 中 cookie 和 会 话 的 原理 及 使 用 方法 。 注 意 这 两 者 的 区 别 ,及 如 何 
安全 有 效 地 使 用 它们 。 


重 点 回 顾 
1. 如 何 设置 .访问 cookie。 
2. 如 何 设置 .访问 会 话 。 
3. 安全 地 使 用 会 话 。 

本 章 实 训 


【 实 训 】 
为 第 9 章 实 训 的 “留言 板 ” 系 统 添加 用 户 管理 功能 ,能 够 注册 、 登 录 及 注销 。 同 时 , 修 
改 留言 板 功能 ,要 求 登 录 后 方 可 留言 ,每 条 留言 能 够 记录 留言 者 信息 。 
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CMS( 内 容 管理 系统 ) 是 PHP 和 MySQL 很 常见 的 应 用 之 一 。 上 一 章 主要 实现 了 
CMS 系统 的 用 户 管理 功能 ,用 户 将 能 够 注册 、 登 录 、 注 销 、 更 改 他 们 的 密码 。 一 旦 用 户 登 
录 , 就 会 使 用 会 话 来 限制 对 页 面 的 访问 ,并 跟踪 用 户 。 

本 章 将 介绍 CMS 的 分 类 和 内 容 管理 的 开发 。CMS 的 内 容 管理 和 分 类 管理 是 必须 具 
备 的 功能 。 栏 目 管理 主要 用 于 创建 用 的 栏目 的 分 类 ,以 区 分 CMS 的 内 容 , 栏 目 管理 包含 
了 栏目 的 创建 修改, 删除。 内 容 管理 可 使 用 第 三 方 的 HTML 编辑 器 来 实现 可 视 化 内 容 
编辑 。 本 章 内 容 将 穿插 引入 类 的 概念 ,并 利用 CI 框架 快速 开发 CMS 常用 的 功能 ,全 章 
通过 循序 渐进 的 手法 引导 PHP 初 学 者 带 着 面向 对 象 的 思想 去 开发 ,培养 开发 者 的 构思 
能 力 。 


11.1 类 与 对 象 


作为 数据 和 功能 代码 的 集合 ,对 象 和 类 是 程序 开发 和 代码 重用 的 基本 单元 。 回 到 上 
一 章 用 户 管理 功能 。 对 于 每 一 个 用 户 ,我 们 需要 跟踪 的 信息 类 型 是 相同 的 ,而 且 每 个 用 户 
的 数据 结构 及 其 调用 的 函数 也 是 相同 的 。 在 编写 程序 时 ,我 们 可 以 决定 用 户 的 字段 及 可 
调用 的 函数 。 用 面向 对 象 程序 开发 (OOP) 的 术语 来 说 ,我 们 在 设计 User 类 。 一 个 类 就 
是 创建 对 象 的 一 个 模板 。 

1. 类 的 声明 

声明 类 的 关键 字 为 class。 建 立 一 个 类 很 简单 。 


<?php 
class my_class{} 


类 到 底 要 干什么 呢 ? 可 以 把 类 看 成 是 一 个 具有 某 一 类 功能 的 黑匣子 ,在 这 里 称 它 为 
一 个 独立 的 整体 。 我 们 只 知道 类 名 ,而 不 知道 里 面 有 什么 东西 。 那 么 ,该 如 何 使 用 这 个 类 
呢 ? 首先 ,要 知道 它 里 面 是 否定 义 了 公共 的 变量 一 一 称 为 属性 ”。 其 次 ,要 知道 它 里 面 定 
义 了 什么 函数 一 一 称 为 “方法 ”。 
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类 名 不 区 分 大 小 写 ,命名 规则 与 PHP 标识 符 的 命名 规则 相同 。 
类 中 的 如 何 定义 公共 变量 呢 ? 很 简单 ,下面 来 扩充 my_class 类 : 
<?Pphp 
class my_class{ 
Var $username; 
?> 
上 面 的 代码 中 定义 了 一 个 公共 的 变量 ,只 是 用 “var 十 空格 十 普通 变量 名 ”的 形式 。 在 
此 基础 上 再 定义 一 个 打印 $ username 的 函数 : 


<?php 
class my _class{ 
Var $username= "CMS"; 
function show_ username ($ username) 
{ 
echo $username; 
} 
} 
2 


2. 创建 对 象 
对 象 是 类 的 实例 。 类 似 于 ,程序 只 定义 了 一 个 用 户 类 ,但 可 以 用 来 创建 多 个 不 同 的 用 
户 。 创 建 对 象 比 定义 对 象 类 简单 。 使 用 关键 字 new 来 创建 一 个 类 的 对 象 。 例 如 ,前 面 已 
定义 了 一 个 my_class 类 ,那么 可 以 这 样 创建 一 个 my_class 对 象 : 
<?php 
$ Name=new my_class (); 
?> 
这 样 就 初始 化 了 上 面 的 my_class 类 ,并 把 这 个 对 象 赋 给 变量 $ Name。 
一 旦 创建 了 一 个 对 象 ,可 以 使 用 一 二 符号 来 访问 对 象 的 属性 和 方法 。 例 如 ,使 用 类 中 
的 函数 执行 打印 操作 : 
<?php 
$Name- > show_username ("教学 CMS"); 
?> 
这 段 代 码 执行 后 将 打印 出 "教学 CMS" 这 个 字符 串 。 所 以 ,类 的 使 用 并 没有 想象 中 
那么 难 。 


11.2 什么 是 Codelgniter(CI) 


毫 无 疑问 , Web 框架 技术 在 近 几 年 得 到 了 突飞猛进 的 发 展 和 普及 ,在 过 去 几 年 里 , 框 
架 技术 的 普遍 经 历 了 比较 大 的 完善 过 程 ,很 大 一 部 分 可 以 归 因 于 Ruby on Rails, 以 及 在 
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其 他 编程 语言 中 流露 出 的 MVC 框架 思想 。 

框架 就 是 为 重用 而 发 明 的 ,存放 在 独立 的 文件 中 ,用 来 简化 重复 操作 的 代码 。 好 的 框 
架设 计 能 实现 需要 的 功能 ,而 且 尽 可 能 地 不 互相 牵连 。 一 个 好 框架 为 开发 者 实现 各 种 功 
,并 且 能 提供 一 步 一 步 的 编程 指导 。 
PHP 作为 网 络 开 发 的 强大 语言 之 一 ,具有 开放 源 代码 、 跨 平台 性 强 、 开 发 快捷 、 效 率 
` 面 向 对 象 , 专 业 专注 等 诸多 优点 。PHP 的 程序 员 早 已 习惯 了 将 需要 重复 使 用 的 代码 
写 在 函数 中 ,并 将 这 些 函数 放 在 include 文件 里 。 同 时 ,各 种 PHP 开发 框架 也 让 程序 开 
发 变 得 简单 有 效 。 

框架 就 是 通过 提供 一 个 开发 Web 程序 的 基本 架构 ,PHP 开发 框架 把 PHPWeb 程序 
开发 放 到 了 流水 线 上 。 换 句 话说 ,PHP teal pet dn 文 节约 了 开 
发 时 间 , 有 助 于 创建 更 为 稳定 的 程序 ,并 减少 开发 者 的 重复 编写 代码 的 劳 这 些 框架 还 
通过 确保 正确 的 数据 库 操作 以 及 只 在 表现 层 编 程 的 方式 帮助 初 Ppt ee 。 
PHP 开发 框架 使 得 开发 者 可 以 花 更 多 的 时 间 去 创造 真正 的 Web 程序 ,而 不 是 编写 重复 
性 的 代码 。 

随 着 框架 技术 的 流行 , 越 来 越 多 的 PHP 框架 涌现 出 来 。 其 中 CI 框架 就 是 较 老牌 的 

一 个 框架 , 它 始 终 是 全 球 排行 前 十 的 框架 。 

CodeIgniter( 简 称 CD) 是 一 套 给 PHP 网 站 开发 者 使 用 的 应 用 程序 开发 框架 和 工具 
包 。 它 提供 一 套 丰 富 的 标准 库 以 及 简单 的 接口 和 J 逻辑 结构 ,其 目的 是 使 开发 人 员 更 快速 
地 进行 项 目 开 发 。 使 用 Codelgniter 可 以 减少 代码 的 编写 量 , 并 将 精力 投入 到 项 目的 创 
造 性 开发 上 。 


11.2.1 下 载 与 安装 CI 


是 完全 免费 的 ,部 署 与 开发 环境 与 前 十 章 一 致 。 访 问 Codelgniter 网 站 : http:// 
www. Ms com/ 可 下 载 到 最 新 版 的 框架 ,目前 的 最 新 版 是 2. 1. 4( 截 至 2014 年 5 
月 ), 其 压缩 包 仅 2.2MB。 官 网 下 载 界 面 如 图 11-1 所 示 。 


融 


ii 到 


Gadalgniter 是 开放 源 代 码 的 
助 你 编写 “优雅 ”PHP 程序 的 


图 11-1 CI 官网 下 载 界面 


将 下 载 的 压缩 包 解 压缩 至 网 站 根 目 录 ,. 会 产生 如 图 11-2 所 示 的 文件 目录 结构 。 

其 中 包含 了 一 个 英文 版 的 用 户 手册 (在 user_guide 文件 夹 中 )。 当 这 些 文件 保存 在 
机 器 上 的 时 候 , 通 过 URL,http://127.0. 0.1 来 访问 它们 。 在 本 例 中 ,CI 将 被 解压 缩 在 
Web 服务 器 根 目录 的 ci 文件 夹 中 。 因 此 ,访问 地 址 为 http://127. 0.0.1/ci 或 http:// 
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localhost/ci。CI 的 默认 首页 如 图 11-3 所 示 。 


出 application 2013-07-29 15:54 ”文件 赤 

轴 system 2013-07-29 15:54 文件 夫 

BB user guide 2013-07-29 15:54 ”文件 夫 
.gitignore 2013-07-29 15:54 GITIGNORE 文 件 
Dtravisyml 2013-07-29 15:54 。 YML 文 件 

国 indexphp 2013-07-29 15:54 ”PHP 文件 

四 licensebt 2013-07-29 15:54 ”文本 文档 


11-2 CI 的 文件 结构 


Ba [hepocalhosyc/ DP- Ox] weicome to codelgriter x 分 安 强 


Welcome to Codelgniter! 


The page you are looking at is being generated dynamically by Codelgniter 
If you would like to edit this page you'l find i located at- 


application/views/welcome_message.php 


The corresponding controller for this page is found at- 


application/controllers/welcome.php 


If you are exploring Codelgniter for the very first time, you should start by reading the User Guide 


Page rendered mn 0.0346 seconds 


图 11-3 CI 的 默认 首页 


11.2.2 CI 文件 结构 


根 目录 目录 结构 如 图 11-2 所 示 。 这 些 文件 夹 可 分 为 三 个 部 分 : 

application 是 项 目 存放 文件 的 目录 。 控 制 器 、 模 型 和 视图 ,全 部 在 application 文 
件 夹 中 。 初 始 情 况 下 除了 默认 首页 的 welcome 视图 和 控制 器 ,其 他 文件 夹 大 多 是 
空 的 或 包含 了 空 文件 。 

在 system 文件 夹 中 的 文件 是 CI 本 身 的 代码 。 如 果 有 兴趣 的 话 ,可 以 研读 它们 ， 
或 者 改变 它们 ,不 过 要 等 了 解 了 CI 是 如 何 工作 之 后 ,才能 这 么 做 。 

有 一 些 文件 夹 中 已 包含 了 文件 ,如 language、config、errors。 在 开发 的 过 程 中 ,可 
能 需要 增加 或 修改 。 


11.2.3 MVC 模式 


CI 实现 了 模型 一 视图 一 控制 器 (Model-View-Controller, MVC) 模 式 。MVC 指 的 是 


一 个 动态 网 站 的 组 织 方 法 。 该 设计 模式 是 1979 年 由 挪威 人 Trygve Reenskaug 首次 提出 
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来 的 ,主要 包括 : 
。 模型 是 包含 数据 的 对 象 ,它们 与 数据 库 交 互 , 对 这 些 数据 进行 存 取 ,使 其 在 不 同 的 
阶段 包含 不 同 的 值 ,不 同 的 值 代表 了 不 同 的 状态 ,具有 特定 的 含义 。 
。 视图 显示 模型 的 状态 ,负责 显示 数据 给 使 用 
者 。 虽然 视图 通常 是 HMTL 页 面 效 果 , 但 视 
图 可 能 是 任何 形式 的 接口 。 比 如 ,智能 手机 
屏幕 或 平板 电脑 。 
。 控制 器 用 来 改变 模型 的 状态 .操作 模型 ,提供 
动态 的 数据 给 视图 。 
MVC 模式 三 元 素 的 交互 示意 图 , 如 图 11-4 
所 示 。 
CI 中 模型 .视图 和 控制 器 文件 都 有 自己 的 文件 图 11-4 MVC 模式 
夹 。 当 安装 CI 的 时 候 ,application 文件 夹 中 包括 名 
models、views 和 controllers 的 子 文件 来。 每 个 CI 网 站 都 包含 这 三 个 文件 夹 ,它们 分 别 
对 应 了 MVC 模式 的 模型 .视图 和 控制 器 。 


11.2.4 应 用 程序 流程 


整个 CI 网 站 是 动态 的 ,可 能 找 不 到 制作 “静态 ”网 页 的 简单 HTML 代码 。 在 根 目录 
中 的 index. php 是 所 有 访问 路 径 的 入 口 。 所 有 的 连接 请 求 被 index. php 文件 拦截 并 进行 
处 理 , 作 用 就 像 一 个 “路 由 器 ”。 换 句 话说 , 它 调用 一 个 “控制 器 ”, 然 后 返回 一 个 “视图 ”。 

“路 由 器 "怎么 知道 调用 哪个 控制 器 ? 如 果 用 户 正在 以 正确 的 URL 请 求 CI 网 站 上 
的 页 面 ,其 URL 一 般 看 起 来 像 这 样 : 


http://example.com/news/latest/10 


仔细 观察 此 URL 地 址 ,可 以 猜测 它 所 完成 的 任务 : 存在 某 个 类 名 为 news 的 控制 器 ， 
调用 此 类 下 的 latest 方法 用 来 提取 10 条 最 新 新 闻 , 然 后 解析 显示 在 最 终 浏览 嚣 页面 上 。 
在 基于 MVC 架构 思想 的 应 用 程序 中 ,经 常会 见 到 如 下 典型 URL 格式 : 


http://localhost/ci/index.php/class/function/ID 


其 中 对 应 的 含义 为 : 

网 站 的 根 目 录 /index. php/[ 控 制 器 类 名 ]/[ 控 制 器 方法 名 ]/[ 所 需 参数 ] 

在 实际 项 目 中 ,以 上 典型 格式 可 能 会 存在 变化 并 趋 于 复杂 。 图 11-5 说 明了 数据 流 如 
何 贯 穿 整个 系统 。 

(1) index. php 作为 前 端 控制 器 ,初始 化 运行 CI 所 需要 的 基本 资源 。 

(2) Router 检查 HTTP 请 求 , 以 确定 谁 来 处 理 请 求 。 

(3) 如 果 缓 存 (Cache) 文 件 存在 , 它 将 绕 过 通常 的 系统 执行 顺序 ,被 直接 发 送 给 浏 
览 器 。 

(4) 安全 (Security)。 应 用 程序 控制 器 (Application Controller) 装 载 之 前 ,HTTP 请 
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图 11-5 应 用 程序 流程 图 


求 和 任何 用 户 提交 的 数据 将 被 过 滤 。 

(5) 控制 器 (Controller) 装 载 模型 .核心 库 、 辅 助 函数 ,以 及 任何 处 理 特定 请 求 所 需 的 
其 他 资源 。 

(6) 最 终 视图 (View) 泻 染发 送 到 Web 浏览 器 中 的 内 容 。 如 果 开 启 缓存 (Caching)， 
视图 首先 被 缓存 ,所 以 将 可 用 于 以 后 的 请 求 。 


11.3 ”C1 的 语法 规则 


在 开始 训练 项 目 之 前 , 先 简单 的 归纳 一 下 CI 的 语法 规则 。 框 架 希 望 文件 按 一 定 的 规 
则 设置 ,否则 它 可 能 无 法 准确 地 定位 和 使 用 它们 。 


11.3.1 控制 器 


一 个 控制 器 就 是 一 个 类 文件 ,是 以 一 种 能 够 和 URI 关联 在 一 起 的 方式 来 命名 的 。 它 
由 URL 直接 调用 ,例如 ,http://www. example. com/index. php/start/hello。 在 上 面 的 
例子 中 ,Codelgniter 将 尝试 寻找 并 装载 一 个 名 为 start. php 的 控制 器 。 控 制 器 使 用 方法 
名 来 调用 方法 ,如 helloO 〇 ;不 过 ,不 能 够 在 一 个 控制 器 内 调用 其 他 控制 器 内 的 方法 。 

控制 器 用 如 下 格式 定义 : 

class Start extends controller 
其 中 控制 器 名 称 的 首 字 母 必须 大 写 , 例 如 这 里 的 Start 控制 器 。 控 制 器 保存 为 . php 文件 ， 
位 于 /application/controllers 文件 夹 中 。 文 件 名 的 首 字母 不 需要 大 写 ,应 该 是 start. php 
而 不 是 Start. php。 还 有 ,在 构造 函数 中 至 少 要 有 如 下 内 容 : 

function display() 

{parent: :Controller ();} 


所 有 的 其 他 代码 一 定 要 写 在 不 同 的 函数 中 ,例如 .hello() 函 数 。 
11.3.2 视图 


视图 是 包含 HTML 代码 的 , 带 有 . php 后 级 的 文件 。 可 在 控制 器 中 使 用 如 下 命令 
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装载 : 
$this->load- >view('testview', $data) 


这 条 命令 将 载 人 并 使 用 视图 。 视 图 保存 为 . php 文件 .位 于 文件 夹 /application/views 中 。 


11.3.3 传递 数据 到 视图 


1. 为 控制 器 添加 逻辑 结构 

首先 ,需要 创建 一 个 可 以 处 理 静 态 内 容 请 求 的 控制 器 类 。 控 制 器 是 一 个 用 来 代 
理 完成 某 项 任务 的 PHP 类 , 它 充当 基于 MVC 架构 应 用 程序 的 “ 粘 合 剂 ”, 控 制 器 会 
成 为 网 站 程序 请 求 的 中 心 。 在 非常 技术 性 的 Codelgniter 的 讨论 中 ,会 把 它 称 为 “ 超 
级 对 象 ”。 

下 面 将 编写 我 们 的 第 一 个 控制 器 Pages, 这 个 控制 器 必须 完成 如 下 几 件 事 : 

。 调用 视图 。 

。 将 基本 URL 信息 的 名 称 传递 给 视图 。 

。 把 另 一 些 数据 传递 给 视图 : 它 正在 期 待 标题 ($ page_title) 和 一 些 文本 ( $ title)。 

(1) 在 文本 编辑 器 或 IDE 中 打开 pages. php。 


脚本 11-1 pages. php 控制 器 ,位 于 /application/controllers。 


1 <?php 

2 class Pages extends CI_Controller { 

3 public function view($ page= 'home'){ 

4 $data['page_title']=" 我 的 第 一 个 CI 程序 "; 
5 $data['title']="CMS 系统 "; 

6 

要 $this->load- >view('header', $data); 
8 $this->load- >view($ page, $data); 

9 $this->load- >view('footer', $data); 
10 

11 } 

2 ?> 


(2) 在 /application/controllers 目录 中 ,创建 一 个 pages. php 的 控制 器 ,其 中 创建 一 
个 名 为 Pages 的 类 。 


class Pages extends CI_Controller 


这 个 Pages 类 继承 了 CI_Controller 类 。 这 就 意味 着 这 个 新 的 pages 类 可 以 继承 CIL_ 
Controller (system/core/Controller. php) 类 里 面 定 义 的 方法 和 变量 。 

(3) 在 Pages 类 包含 一 个 名 为 view 的 方法 ,这 个 方法 里 定义 了 一 个 参数 , 它 的 值 是 
即将 加 载 的 页 面 的 名 称 。 


public function view(5page= 'home') 
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(4) 在 view 方法 中 定义 了 名 为 $page_title 和 $ title 的 两 个 数组 变量 ,用 来 显示 于 
页 面 标题 与 正文 标题 。 

$data['page title']=" 我 的 第 一 个 cI 程序 "; 

$data['title']="CMS 系统 "; 

(5) 按照 需要 显示 的 顺序 来 加 载 那些 视图 。 任 何 php 类 一 样 ,在 自己 的 控制 器 中 ,使 
用 $ this 来 调用 它 , 这 样 就 可 以 实现 用 $ this 来 加 载 所 有 变量 、 视 图 和 对 这 个 框架 进行 
一 般 操作 。$ this 一 二 load 一 二 view0 〇 方法 中 的 第 二 个 参数 是 用 来 传递 值 给 视图 的 。 

$this- >1l0ad- >view('header', $data); 

$this- >load- >view($ page, $data); 

$this- >1l0ad- >view('footer', $data); 

数组 中 的 每 个 值 都 被 定义 成 与 其 关键 字 相 同 的 一 个 变量 ,如 控制 器 中 $ data['title] 
的 值 就 等 同 于 视图 中 变量 $ title。 

2. 传递 值 到 视图 

在 views 文件 夹 中 需要 创建 两 个 基础 的 页 面 模板 ,也 就 是 视图”, 分 别 是 页 面 的 页 头 
(header) 和 页 脚 (footer) 。 

(1) 创建 页 头 文件 /application/yviews/header. php。 


脚本 11-2 header. php 视图 ,位 于 /application/ views。 


1 < !DOCTYPE html PUBLIC "- //W3C//DTD XHTML 1.0 Transitional//EN" 

2 "http://www.w3.org/TR/xhtml1/DTD/xhtmll- transitional .dtd"> 

3 <html xmlns= "http://www.w3.org/1999/xhtml"> 

4 <head> 

5 <meta http- equiv= "content— type" content= "text/htmlL; charset=gb2312" /> 
6 <title><?php echo $ page title;?></title> 

7 <style type= "text/css"> 

8 EE 

9 </style> 

10 < /head> 

11 

12 <body> 

13 <div id="wrapper"> 

14 <div id= "content"> 

15 <div id=- "nav"> 

16 <h3> 请 选择 < /h3> 

17 <ul> 

18 <1i class= "navtop"><a href="#" title= "首页 "> 首页 </a></1i> 
19 <1li><a href="#" title= "注册 "> 注册 </a></1i> 

20 <1li><a href="#" title= "查看 文章 "> 查看 文章 </a></1i> 
21 </al> 

Fe </div> 
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页 头 文件 包括 在 正式 加 载 视图 前 需要 的 基本 的 HTML 代码 .并 省 略 了 部 分 CSS 代 
码 。 同 时 ,这 里 还 输出 了 已 经 在 控制 器 中 定义 好 的 $ page_title 变量 ,作为 网 页 标题 。 现 
在 来 创建 一 个 页 脚 application/views/footer. php。 

(2) 创建 页 脚 文 件 /application/views/footer. php。 


脚本 11-3 footer. php 视图 ,位 于 /application/views。 


</div> 

<div id= "footer"> <p> gcopy; Copyright 2014< /p>< /div> 
3 </div> 

4 </body> 

5 </htm> 


(3) 创建 主体 文件 /application/views/home. php。 


脚本 11-4 home. php 视图 ,位 于 /application/views。 


1 <hl><?Php echo $title;?>< /hl> 

FE <h2 id= "mainhead"> 父 标题 < /hl> 
3 <p> 主 要 内 容 < /p> 

4 <p> 主 要 内 容 < /p> 

5 <h2> 子 标题 < /h2> 

6 <p> 主 要 内 容 < /p> 

7 <p> 主 要 内 容 < /p> 


在 页 头 模板 (header. php) 中 ,用 $ page_title 变量 来 自 定 义 页 面 标题 (二 title 二 ) ,在 
主体 (home. php) 中 ,用 $title 变量 来 自 定义 文字 标题 (二 hl 之 ) 。 而 这 些 变 量 的 值 , 是 在 
控制 器 Pages 的 view() 方 法 中 定义 的 ,并 将 它们 作为 一 个 元 素 赋 给 了 $ data 数组。 通过 
view() 方 法 中 传递 视图 的 参数 传递 值 给 视图 。 

(4) 这 个 控制 器 现在 可 以 工作 了 ,在 浏览 器 中 输入 地 址 [你 的 网 址 ]index. php/ 
pages/view 就 可 以 看 到 Home 页 面 ,如 图 11-6 所 示 。 

站 eel |) 
ODE -ox [Ssms—+aes x 国 | 分 空 
BE 


图 11-6 ”home 页 面 浏览 效果 


257 


PHP+MySQL 项 目 实例 开发 


参数 home 是 URL 的 最 后 一 部 分 .并 被 传递 给 view 方法 .再 经 由 这 个 方法 传 给 视 
图 。 特 别 注意 的 是 : 视图 一 定 要 与 控制 器 相对 应 。 如 果 视 图 中 并 没 为 一 个 变量 安排 显示 
的 位 置 , 它 将 不 被 显示 。 如 果 视 图 正在 期 待 一 个 变量 ,而 这 个 变量 并 没有 在 控制 器 中 声明 
并 赋值 ,浏览 器 可 能 收 到 一 个 错误 信息 (当然 ,如 果 变 量 被 声明 , 则 不 会 有 错误 信息 ,但 它 
不 会 正常 显示 ) 。 


11.4 项 目 州 练 CMS 系统 设计 与 开发 


在 上 一 节 中 ,通过 如 何 写 出 一 个 包含 静态 页 面 的 类 来 了 解 了 一 些 CI 这 个 框架 的 基本 
概念 ,并 实现 动态 地 把 数据 从 控制 器 传递 给 视图 。 下面 以 创建 一 个 CMS (Content 
Management System 内 容 管理 系统 ) 的 项 目 训 练 来 介绍 动态 内 容 和 如 何 使 用 数据 库 。 


11.4.1 项 目 说 明 


CMS 是 Content Management System 的 缩写 , 意 为 “内 容 管理 系统 ”。 内 容 管理 系统 
是 企业 信息 化 建设 和 电子 政务 的 新 宠 。 对 于 内 容 管理 ,业界 还 没有 一 个 统一 的 定义 ,不 同 
的 机 构 有 不 同 的 理解 。CMS 又 被 称 为 文章 管理 系统 ,大 多 数 的 CMS 主要 包含 新 闻 发 布 
功能 或 文章 发 布 功 能 ,主要 用 于 搭建 新 闻 资讯 类 网 站 。 一 般 商 业 CMS 的 软件 都 包含 前 
台 显示 、 后 台 管理 ,用户 管理 ,文章 管理 ,分 类 管理 等 主要 功能 。 本 项 目的 功能 旨 在 使 用 
CI 框架 快速 开发 一 个 简易 CMS ,要 求 能 够 设计 发 布 文章 的 相关 数据 表 , 能 读 取 文 章 与 添 
加 文章 ,文章 分 类 管理 的 部 分 将 作为 实 训 内容 。 


11.4.2 项 目 原理 


使 用 CI 框架 来 开发 项 目 是 为 了 要 使 PHP 编程 更 容易 和 更 有 生产 力 。 动 态 网 站 难免 
都 会 涉及 数据 库 管 理 的 操作 ,CI 有 它 自 己 专 门 的 数据 库 管理 的 类 ,名 为 Active Record 
类 。Active Record( 以 下 简称 AR) 类 的 价值 是 非常 高 的 , 它 是 提高 生产 力 的 主要 工具 。CI 
保留 了 用 传统 的 方法 编写 数据 库 查询 的 功能 ,但 是 这 里 将 不 会 详细 介绍 这 部 分 内 容 。 使 
用 CI 的 AR 类 后 ,开发 者 可 能 不 会 再 用 传统 的 方式 来 执行 数据 库 查询 了 。 

CI 使 用 的 是 修改 过 的 AR 数据 库 模 式 。 这 种 模式 是 以 较 少 的 程序 代码 来 实现 信息 
在 数据 库 中 的 获取 、 插 入 和 更 改 。 有 时 只 用 一 两 行 的 代码 就 能 完成 对 数据 库 的 操作 。CI 
不 需要 每 一 个 数据 库 表 拥有 自己 的 类 。 它 提供 了 一 个 更 简单 的 接口 。 

不 只 是 简单 ,使 用 AR 的 一 个 主要 的 优点 是 允许 创建 独立 的 数据 库 应 用 程序 ,因为 查 
询 语 法 是 由 数据 库 的 适配器 来 产生 的 。 它 可 以 进行 更 安全 的 查询 ,因为 系统 会 自动 的 对 
所 有 的 输入 值 进 行 转 义 。AR 是 一 个 “设计 模式 ”。 另 一 方面 又 是 一 个 高 度 抽象 的 东西 ， 
就 像 MVC。 它 本 身 不 是 代码 ,只 是 一 个 模式 。 对 于 代码 ,有 一 些 不 同 的 解释 。 它 的 核心 
是 把 数据 库 和 PHP 对 象 建 立 一 个 对 应 关系 。 每 次 当 执 行 一 个 QUERY 语句 。 每 一 张 数 
据 库 里 的 表 是 一 个 类 ,每 一 行 是 一 个 对 象 。 所 有 需要 做 的 只 是 创建 它 ,修改 它 或 者 删除 
它 。 例 如 “方法 ”, 是 从 类 继承 而 来 。Ruby on Rails 是 使 用 AR 模式 的 ,CI 也 是 ,尽管 这 
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两 种 框架 实现 AR 的 方式 有 一 点 不 同 。 

11.4.3 设计 过 程 

1. 创建 数据 表 

在 进行 具体 开发 之 前 ,首先 要 创建 一 张 数据 表 , 用 来 存放 文章 的 相关 记录 。 在 第 9 章 
和 第 10 章 中 使 用 的 mydatabase 数据 库 中 ,新建 一 张 news 数据 表 , 其 中 包含 3 个 字段 ,分 


别 是 id( 文 章 ID) \title( 文 章 标题 ) 及 text( 文 章 正文 ) 。 在 phpmyadmin 中 的 数据 表 结 构 
如 图 11-7 所 示 。 


图 服务 器 : localhost ， 昂 孝 据 库 : mydatabase ， 国 表 : news 
_ 国 水 5% 攻 结构 “ 民 sal 万 接 素 了 括 入 卫 号 由 npert 估量 作 国 下 二 加 呈 陆 
字 恨 。 类型 机 理 。 展 性 Nw 力 认 。 疾 外 各 作 
回 到 int1t) 否 auto incsemet 围 必 X 加 而 了 岗 梧 
口 tle varchar(128) utB_general_d 否 加 于 交加 加 四 癌 
回 text ax utl3_general od 否 国 怀 X 友 而 局 网 
全 选 /全 部 不 选 关内 双 国 X 国 回回 园 
mydatabase (2) 总 打印 预览 出 规划 表 结 构 国 en 
i 屠 洒 加 1 字段 图 于 来 结尾 日 于 来 开头 日 于 ”之 后 区 国 
news 
国 users 已 使 用 空间 和 统计 
键 名 大 型 ”基教 报 作 ”字段 类 型 。 用 法 语句 
PRIMARY PRIMARY 0 PX 孝 据 0 字 节 = 动态 
ri 罕 引 1024 字 节 ua_general_d 
J 纺 计 1024 字 季 ”有数 0 
下 一 个 Automdex 1 
创建 时 间 2014 年 05 月 24 日 1433 
最 后 更 新 时 间 2014 年 05 月 24 日 14:33 
图 11-7 news 表 结 构 
2. 创建 数据 模型 


数据 库 的 运算 并 不 是 在 控制 类 中 进行 的 ,而 是 在 数据 模型 中 ,这 样 就 可 以 在 系统 中 很 
容易 地 被 反复 使 用 。 数 据 模 型 就 是 对 数据 库 或 其 他 数据 存储 方式 进行 查询 、 插 入 和 更 新 
的 地 方 , 它 们 的 功能 是 展示 数据 。 

(1) 数据 库 配置 。 

CI 有 一 个 配置 文件 用 来 存放 数据 库 连 接 信息 ,包括 username: 用 户 名 、password: 密 
码 .database name: 数 据 库 名 等 ,这 个 配置 文件 位 于 /application/config/database. php。 

配件 文件 中 存放 了 如 下 格式 的 一 个 多 维 数组 ,其 中 需要 修改 的 行 及 内 容 如 下 : 


$db['default'] ['hostname']="localhost"; 
$db['default'] ['username']="root"; 
$db['default'] ['password']=""; 

$ db['default'] ['database']= "database name"; 
参数 解析 : 

。 hostname 一 一 数据 库 的 主机 名 ,通常 位 于 本 机 .可 以 表示 为 localhost。 
需要 连接 到 数据 库 的 用 户 名 。 


登录 数据 库 的 密码 。 


» username 


。 password 
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需要 连接 的 数据 库 名 。 
(2) 进入 /application/models 文件 夹 新 建 一 个 文件 news_model. php 。 


。 database 


脚本 11-5 news_moodle. php 数据 模型 ,位 于 /application/models。 


1 <?php 

2 class News_model extends CI Model { 

4 Public function construct(){ 

3 $this->10ad- > database (); 

6 } 

‘ public function get_ news ($ id=FALSE) { 
8 if ($id===FALSE){ 

9 $query=$ this- >db- >get('news'); 
10 return $query-> result array(); 
11 } 

12 


13 $query=$ this- >db- >get where('news', array('id'=>$id)); 
14 return 5 query-> row array(); 


通过 上 面 的 代码 可 以 实现 两 个 不 同 的 查询 ,可 以 得 到 所 有 的 文章 记录 ,也 可 以 通过 
id 得 到 某 一 篇 文章 。$ id 变量 在 查询 前 并 没有 被 检验 过 ,因为 Active Record 类 已 经 把 
这 个 工作 做 完了 。 

3. 显示 文章 

查询 由 数据 模型 news_model 完成 ,下 面 就 要 把 这 个 数据 模型 和 用 来 显示 文章 内 容 
的 视图 联系 起 来 。 这 个 工作 在 之 前 写 的 pages 控制 类 中 就 可 以 实现 ,但 这 里 将 重新 定义 
一 个 新 的 news 控制 类 。 

(1) 进入 /application/controllers 文件 夹 新 建 一 个 文件 news. php。 


1 <?php 

吉 class News extends CI_Controller { 

| public function construct(){ 

4 Parent::_construct (); 

5 $this->load- >model ('news model'); 
6 

7 

8 

» 


public function index(){ 
$data['news']=$ this- >news model->get_news(); 

10 $data['page title']=' 所 有 文章 '; 

11 

12 S$this->load- >view('header', $data); 


13 
14 
15 
16 
17 
18 
19 
20 
21 
22 
23 
24 
25 
26 
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Sthis->load- >view('index', $data); 
Sthis->load- >view('footer'); 


public function view ($id) { 
$data['news item']=$this- >news model- >get news ($id); 
$data['page title']=$data['news item']['title']; 


$ this- >10ad- >view('header', $data); 
$this->load- >view('view', $data); 


$this->10ad- >view('footer'); 


} 


上 面 的 代码 与 之 前 pages 控制 器 很 类 似 。 首先,_construct 方法 是 父 级 类 (CI_ 


Controller) 的 构造 函数 ,并 调用 了 数据 模型 ,这 样 这 个 控制 器 中 的 所 有 方法 就 能 使 用 
news_model 数据 模型 了 。 


public function _construct (){ 


} 


Parent::_construct (); 
$this->1oad- >model ('news_model'); 


其 次 ,News 控制 器 中 有 两 个 方法 分 别 用 来 显示 所 有 的 文字 和 某 一 条 文字 。 通 过 
index 方法 中 的 $ this 一 二 load 一 二 view 方法 把 数据 传递 给 视图 。index 方法 将 从 数据 模 
型 中 获得 所 有 文章 的 记录 ,并 把 它们 赋 给 一 个 变量 。 页 面 的 标题 也 赋 给 了 $ data[ 'page_ 
title] ,这 些 所 有 的 数据 都 会 传递 给 视图 。 


public function index(){ 


} 


$data['news']=$this->news model- >get news(); 
$data['page title']=' 所 有 文章 '; 

$this->load- >view('header', $data); 
$this->lo0ad- >view('index', $data); 


$ this->load- >view('footer'); 


在 第 二 个 方法 view 中 可 以 看 到 $id 变量 被 传递 给 了 数据 模型 中 的 方法 。 数 据 模 型 
就 是 用 这 个 id 来 确定 需要 返回 哪 一 篇 文章 的 。 


public function view($ id){ 


$data['news item']=$this->news model- >get news($id); 
$data['page title']=$data['news item']['title']; 
$this->1lo0ad- >view('header', $data); 

$this->load- >view('view', $data); 

$this->load- >view('footer'); 
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} 
(2) 创建 一 个 视图 来 显示 这 些 文章 。 


脚本 11-6 index. php 视图 ,位 于 /application/views。 


< ?php foreach ($news as Snews_item) : ?> 


1 

3 <h3><?php echo $news item['title'] ?></h3> 

4 <p><a href="http://localhost/ci/index.php/news/view/< ?php echo $news item['id 
'] ?> "> 查看 全 文 </a></p> 

5 

6 “<?Pphp endforeach ?> 


在 这 里 ,每 篇 文章 的 标题 都 被 循环 出 来 展示 给 读者 ,页 面 中 循环 显示 所 有 文章 的 标 
题 , 每 个 标题 还 配 有 “查看 全 文 ” 的 超 链 接 ,链接 至 该 文章 的 详情 页 面 。 

浏览 地 址 为 http://localhost/ci/index. php/news, 即 可 看 到 如 图 11-8 所 示 的 文章 概 
述 页 面 。 


@ hapyV/localhostyciindexphf 及- OX 


大 几 百度 的 开放 适 配 服 务 进行 PC 端 和 手机 端 网 页 对 


i E34 

Web 性 能 压力 测试 工具 Siege 介 绍 
二 看 全 文 

Java 静 态 代 码 块 、 非 静态 代码 块 和 构造 函数 执行 顺序 
查看 全 文 

Linux 下 web 性 能 测试 工具 http_load 介 绍 

可 看 全 文 
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图 11-8 文章 概述 页 面 效果 
(3) 创建 一 个 视图 来 显示 每 一 篇 文章 。 
脚本 11-8 view. php 视图 ,位 于 /application/views。 
<?php 
echo "<h2> ' .Snews item['title'].'</h2>"'; 


1 
2 
3 echo Snews item['text']; 
4 ?> 
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浏览 地 址 : http://localhost/ci/index. php/news/view/1, 即 可 看 到 如 图 11-9 所 示 的 
id 为 1 的 文章 详情 页 面 。 


Gas 


利用 下 记 的 并 让 滞 孔 服务 济 生 PC 沉 和 
手机 端 网 页 对 应 
百度 的 开放 话 配 服务 【 终 训 和 本 服务 如果 


拥有 PC 沾 和 手机 站 ， 
够 在 内 容 上 对 应 ， 您 可 向 百度 “提交 ”PC 页 一 了 应 关系 ， 才 和 将 有 助 于 百度 在 
移动 搜索 中 将 原 PC 页 结果 普 换 为 对 应 的 手机 页 结果 


ES 
文章 
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图 11-9 某 一 篇 文章 详情 页 面 效 果 


4. 添加 文章 

前 面 通过 CI 从 数据 库 中 读 取 数据 ,下 面 将 介绍 通过 怎么 扩展 之 前 写 过 的 news 控制 
器 和 数据 模型 来 实现 如 何 向 数据 库 中 写 人 数据 的 功能 。 

1) 创建 表单 

为 了 向 数据 库 里 输入 数据 ,需要 创建 一 个 表单 来 输入 需要 被 存储 的 信息 。 在 这 个 项 
目 中 将 需要 一 个 至 少 带 两 个 输入 域 的 表单 ,一 个 用 来 输入 标题 ,一 个 用 来 输入 内 容 。 

在 /application/views 文件 夹 新 建 一 个 文件 create. php ,创建 一 个 新 的 视图 (参见 脚 
本 11-8) 。 


脚本 11-8 create. php 视图 ,位 于 /application/views。 


<h2> 添 加 一 篇 新 文章 < /h2> 
< ?php echo validation errors(); ?> 
< ?php echo form open ('news/create')?> 


<label for= "title"> 文 章 标题 < /label> 

<input type= "input" name="title" /><br /> 

<label for="text"> 文 章 正文 </label> 

<textarea name= "text">< /textarea> <br /> 

<input type= "submit" name= "submit" value= "添加 文章 " /> 


oooaonnwDNpr- 


叫 
了 


11 </fom> 
里 只 有 两 个 不 熟悉 的 新 内 容 : 一 个 是 form_open() 函 数 , 另 一 个 是 validation_ 
errors() 函数 。 
form_open() 由 CI 的 表单 辅助 函数 提供 ,用 来 提供 表单 元 素 和 一 些 额 外 功能 ,例如 添 


263 


264 


PHP+MySQL 项 目 实例 开发 


加 隐藏 的 安全 类 。validation_errors() 用 来 报告 表单 验证 中 出 现 的 错误 信息 , 它 将 返回 验 
证 器 送 回 的 所 有 错误 信息 。 如 果 没有 错误 信息 , 它 将 返回 空 字符 串 。 

2) 在 控制 器 中 添加 方法 

回 到 News 控制 器 ,在 这 里 需要 做 两 件 事 : 一 件 是 检查 表单 是 否 被 提交 了 , 另 一 件 是 
检查 提交 的 数据 是 否 能 够 通过 验证 规则 。 这 里 需要 用 到 表单 验证 库 来 做 这 些 。 

打开 之 前 /application/controllers/news. php, 加 入 脚本 11-9。 


脚本 11-9 news. php 控制 器 ,位 于 /application/controllers。 


public function create(){ 
$this- >10ad- >helper('form'); 
$this- >1lo0ad- >library('form validation'); 


$this- >form validation- >set_ rules('title', 'Title', 'required'); 


1 

尝 

有 

4 

5 $data['page_title']=' 添 加 一 篇 新 文章 '; 

6 

7 

8 $this- >form validation- >set_ rules('text', 'text', 'required'); 


10 if($ this- > form validation- >run()===FALSE){ 


11 S$this->load- >view('header', $data); 
越 $ this->load- >view('create'); 

把 $ this->1load- >view('footer'); 

14 }else{ 

5 $this- >news model- > set_news () 7 

16 $ this- >load- >view('success'); 

17 } 

18 } 


上 面 的 create 方法 将 添加 至 News 类 中 .前 几 行 载 入 了 表单 辅助 函数 和 表单 验证 库 ， 
这 样 ,表单 验证 的 规则 就 被 设 定好 了 。set_rules() 方 法 包含 三 个 参数 ,第 一 个 是 输入 域 的 
名 称 ,第 二 个 是 错误 信息 的 名 称 , 第 三 个 是 错误 信息 的 规则 一 一 这 里 的 规则 是 输入 内 容 的 
文本 域 必 填 。 

$this->form validation- >set rules('title', 'Title', 'required'); 

$ this->form validation- > set_ rules('text', 'text', 'required'); 

正如 create 方法 所 展示 的 ,CI 拥有 一 个 强大 的 表单 验证 库 。 读 者 可 以 从 CI 的 用 户 
手册 中 了 人 解 到 这 个 库 的 更 多 内 容 。 

在 验证 完 表单 数据 后 ,有 一 个 用 来 检查 表单 验证 是 否 运 行 成 功 的 条 件 。 如 果 没 有 成 
功 ,显示 表单 ,如 果 提 交 成 功 并 且 通 过 了 验证 , 则 会 调用 数据 模型 。 这 之 后 会 加 载 一 个 显 
示 成 功 信息 的 视图 。 在 这 里 application/view/success. php 创建 一 个 新 的 视图 用 来 显示 
成 功 信息 。 

浏览 地 址 为 http://localhost/ci/index. php/news/create. 即 可 看 到 如 图 11-10 所 示 
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的 添加 新 文章 的 页 面 。 
PF leis | 
Ge 华 TipyVocahosyayndexph P -OX] 名 二 In_ 拓 新 广 间 x 辆 | 分 妆 澡 


添加 一 篇 新 文章 


文章 标题 


文章 正文 


图 11-10 ”添加 文章 页 面 效果 
3) 完善 数据 模型 
最 后 一 件 工作 ,就 是 写 一 个 方法 用 来 向 数据 库 中 写 和 数据。 这 里 将 用 AR 类 来 插入 
信息 ,并 用 到 输入 类 来 获得 post 数据 。 打开 之 前 创建 的 news_model 数据 模型 加 入 脚 
本 11-10。 


脚本 11-10 news_moodle. php 数据 模型 ,位 于 /application/models。 


1 public function set news(){ 

2 $this- >1lo0ad- >helper('url'); 

3 

4 $data=array( 

5 'title'=>$this->input->post('title’), 
6 'text'=>$ this->input- >post('text') 

量 ) 7 

8 return $ this- >db- > insert ('news', $data); 
9 上 


这 个 set_news 方法 是 用 来 向 数据 库 插入 文章 条 目的 。 先 准备 好 向 $ data 数组 输入 
的 记录 。 这 里 的 每 个 元 素 都 对 应 着 早 前 创建 的 数据 表 中 的 每 一 列 。 这 里 有 个 新 的 方法 叫 
post() , 它 是 由 输入 类 提供 的 。 这 个 方法 可 以 确保 数据 是 被 过 滤 过 的 .从 而 保护 服务 器 
不 被 其 他 人 恶意 攻击 。 这 个 输入 类 是 默认 加 载 的 。 最 后 ,就 是 将 $ data 数组 插入 到 数 
据 库 。 

由 此 ,完成 了 添加 文章 的 功能 ,在 添加 文章 的 表单 中 输入 一 篇 新 文章 的 内 容 , 成 功 后 
将 转 至 success 视图 ,如 图 11-11 所 示 。 

5. 调整 链接 

目前 为 止 ,CMS 项 目的 功能 已 基本 实现 .但 右 侧 导航 部 分 的 链接 依然 欠 妥 。 由 于 CI 
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【< OE ET x* 国 | 分 安 剖 


添加 文章 成 功 ! 
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图 11-11 添加 文章 成 功 页 面 效 果 


框架 中 的 URL 地 址 是 基于 MVC 架构 思想 的 ,所 以 不 能 简单 地 通过 文件 地 址 实现 超 链 
接 。 因 此 ,这 里 将 启用 base_url 这 个 辅助 类 。 

1) 修改 整 站 的 base_url 

这 个 配置 文件 位 于 /application/config/config. php。 配 件 文件 中 存放 了 如 下 格式 的 
一 个 多 维 数组 ,其 中 需要 修改 的 行 及 内 容 如 下 : 

$config['base url']= "7 

在 这 里 ,可 以 指定 根 URL 的 路 径 。 例 如 ,在 本 例 中 可 指定 为 : 

$ config['base url']= 'http://localhost/ci/ "7 

2) 调整 导航 模板 视图 header. php 

<ul> 

<1i class= "navtop"><a href= "<?=base_url("index.php/news");?>" title=" 首 页 "> 首页 < /a 

></1i> 

<1i><a href="<?=base_url("index.php/news/create");?2>"”title= "添加 文章 "> 添加 文章 </a 

></1i> 


<1i><a href= "<?=base url ("index.php/news");?>" title= "查看 文章 "> 查看 文章 </a></1i> 
</ul> 


将 导航 区 域 的 列表 标签 中 的 超 链接 标签 修改 URL 地 址 ,设置 为 base_url() 的 格式 ， 
当 将 URI 段 作 为 参数 传 给 这 个 函数 时 ,index. php 文件 名 会 被 加 到 URL 后 面 。 
例如 ,修改 为 : 


<a href="<?=base_url("index.php/news");?>" title=" 首 页 "> 首页 < /a> 
将 返回 的 HTML 代码 为 : 

<a href= "http://localhost/ci/index.php/news" title= "首页 "> 首页 </a> 

修改 完成 后 的 首页 效果 ,如 图 11-12 所 示 。 
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@ htpy/localhoscfinde P ~ OX | Gris x 因 | 介 交 侣 


查看 全 文 
Web 性 能 压力 测试 工具 Siege 介 绍 
查看 全 文 
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Linux 下 web 性 能 测试 工具 http_load 介 绍 

查看 全文 
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11-12 包含 有 效 链接 的 页 面 效 果 


读者 还 可 通过 . htaccess 文件 的 目录 改变 配置 方法 ,去 掉 URL 中 的 index. php ,并 归 
类 站 点 中 的 CSS 和 JS 文件 ,设置 相关 访问 权限 ,以 获得 更 美观 方便 的 URL。 


本 章 小 结 


本 章 首先 介绍 了 PHP 中 类 和 对 象 的 概念 及 使 用 方法 .然后 结合 了 一 个 PHP 知名 的 
开源 框架 一 一 CodeIgniter, 详 细 介绍 了 一 个 内 容 管理 系统 (CMS) 的 制作 过 程 。 


重点 回顾 


1. 如 何 声明 类 及 使 用 对 象 。 
2. CI 框架 的 部 署 与 MVC 模式 。 
3. 如 何 使 用 CI 框架 实现 动态 网 站 的 开发 。 


本 章 实 训 


使 用 CI 框架 ,为 本 章 的 CMS 系统 添加 文章 分 类 管理 功能 。 
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